mirror of
				https://github.com/paperless-ngx/paperless-ngx.git
				synced 2025-10-24 03:26:11 -05:00 
			
		
		
		
	Compare commits
	
		
			469 Commits
		
	
	
		
			v2.15.0-be
			...
			fb9200a344
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fb9200a344 | ||
|   | 8d1f23e9d6 | ||
|   | c8850fa752 | ||
|   | 19a54b3b23 | ||
|   | 1cdd8d9ba8 | ||
|   | 4449dbadb5 | ||
|   | 43b4f36026 | ||
|   | 0e35acaef5 | ||
|   | 19ff339804 | ||
|   | 6b868a5ecb | ||
|   | 6231211f9b | ||
|   | 6dbd32759d | ||
|   | e0512e35a2 | ||
|   | 4cff907ba0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4b32c3228e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4ddac79f0f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d4be3bd31d | ||
|   | d5aba09de9 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | f2ef9af291 | ||
|   | 4905edbf79 | ||
|   | feb5d534b5 | ||
|   | d230514dd3 | ||
|   | 1709aee903 | ||
|   | 3e4aa87cc5 | ||
|   | fc95d42b35 | ||
|   | c4346124c3 | ||
|   | 44b8c4881a | ||
|   | d3d8eef0b6 | ||
|   | a283c1c320 | ||
|   | f3220ce981 | ||
|   | 2dc4f1f49b | ||
|   | 17509171bb | ||
|   | 9e11e7fd05 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 84942a4e69 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 48168df320 | ||
|   | cec665f8d5 | ||
|   | 8adc26e09d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 84d85d7a23 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 71f20f62d0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a94a8e4c6f | ||
|   | 7a1aae7749 | ||
|   | 894939e492 | ||
|   | f431578f43 | ||
|   | 1b18c14188 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | d721a88a2f | ||
|   | f7b4d38e39 | ||
|   | 46cf6b4583 | ||
|   | 2d701c5c1b | ||
|   | 1123d845ec | ||
|   | dfa6308ca4 | ||
|   | b5a17a8d11 | ||
|   | cfac74319f | ||
|   | f9f069b092 | ||
|   | b2703b4605 | ||
|   | 852eb0ef36 | ||
|   | 0870d42eae | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e2cf95f8af | ||
|   | a79c8dc51c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4b95c2f0e5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e1c8cd779b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cc7c7f31ba | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1d30ce2afa | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 5aa86f8755 | ||
|   | de2ddad5ee | ||
|   | d2064a2535 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | cc621cf729 | ||
|   | fc4134e15c | ||
|   | ac1b420966 | ||
|   | 80595899c1 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 9463a8fd26 | ||
|   | 58ab137282 | ||
|   | 05c216b2a8 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | d6db2d3fce | ||
|   | a6e41b4145 | ||
|   | cb927c5b22 | ||
|   | 107374af71 | ||
|   | a77141e133 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 117dfb83fe | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fdef774a16 | ||
|   | 08887cb8e3 | ||
|   | 7b679e11bc | ||
|   | dbbebaeb89 | ||
|   | d9459ac37f | ||
|   | 4e0f5dff95 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 10ccccc987 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 27d72ebb18 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 909ccebb34 | ||
|   | 4275e18c10 | ||
|   | 0088333360 | ||
|   | ed1d488d6e | ||
|   | b25b15ba32 | ||
|   | f2fabc81d4 | ||
|   | f94c3eeea8 | ||
|   | bf468ac64f | ||
|   | 22064ed004 | ||
|   | 23daa0b974 | ||
|   | 7b63f5a98c | ||
|   | 7c76377477 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 56c70bf177 | ||
|   | daf47f377b | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 64f31cac0c | ||
|   | dcc503c35f | ||
|   | a583cff21c | ||
|   | bfd468103b | ||
|   | be0c1fd1ed | ||
|   | 82370963da | ||
|   | 0fdfa42a83 | ||
|   | 0f0ba92e15 | ||
|   | 5f0281e427 | ||
|   | a0c7785881 | ||
|   | 349fbce579 | ||
|   | 217b004884 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 29c36542fa | ||
|   | d5b87aeffb | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 9225a38458 | ||
|   | 3fa89b85d7 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 5e6b49971f | ||
|   | be63c79db1 | ||
|   | 26c70b69c4 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | e0b0dd8548 | ||
|   | 1bbac9948a | ||
|   | ca9b5d9586 | ||
|   | 521fd1c957 | ||
|   | f00a565130 | ||
|   | d878bc153a | ||
|   | f5e6951910 | ||
|   | 91102d0335 | ||
|   | 82ec1be622 | ||
|   | 01a8cf6f36 | ||
|   | 6bdb365f87 | ||
|   | 696e591a3b | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 3c2782c3a9 | ||
|   | a68800d53c | ||
|   | 52a937cdcc | ||
|   | 00e629d957 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 243b3bc812 | ||
|   | 0ccc2da9bb | ||
|   | b6dbbec019 | ||
|   | b1c406680f | ||
|   | 42bdbc1b2d | ||
|   | 2f529a9500 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ee6b700243 | ||
|   | b1a84c65ed | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | edb8c06e2a | ||
|   | 1b6ec65f6e | ||
|   | 6d72ee795f | ||
|   | 6730896894 | ||
|   | 5d6ea70434 | ||
|   | fac1ee4283 | ||
|   | 1bee1495cf | ||
|   | 6dca4daea5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 54e2b916e6 | ||
|   | ea62e30c90 | ||
|   | 91511b45cd | ||
|   | b5dd751b67 | ||
|   | 07c298523a | ||
|   | 0ea159683d | ||
|   | f0b6e79d14 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 302cb22ec6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 4210addb46 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 06746b4b31 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2f5533a179 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2f267341f8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 88befee527 | ||
|   | c4a7186cd2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d974f092aa | ||
|   | 23501b9060 | ||
|   | f09965464a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | ae5bd2d2fd | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2b73007e7e | ||
|   | 8505fa3e54 | ||
|   | a51093afc2 | ||
|   | 2fdae59288 | ||
|   | 4637f5c5e5 | ||
|   | 5e7ee924ff | ||
|   | fded55dc70 | ||
|   | 20da51278e | ||
|   | 293c84d871 | ||
|   | 1fe8599266 | ||
|   | 5410074062 | ||
|   | 4b8f6ed643 | ||
|   | f8689c4819 | ||
|   | cebc227701 | ||
|   | 814df94e8d | ||
|   | fa496dfc8d | ||
|   | 924471b59c | ||
|   | feb320cae9 | ||
|   | 9178af5fb2 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 850444c2fc | ||
|   | 90baba2cec | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 9889c59d3d | ||
|   | 3d2a3ede71 | ||
|   | bc019fab96 | ||
|   | 1cd21d0f38 | ||
|   | f940ed0b7b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3180ccf4cb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 43abb0541b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a3a405354f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 09e98d600e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 01a39b9bb4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3b0b40f071 | ||
|   | 6dce83865f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 18252a19d7 | ||
|   | 733a9674d6 | ||
|   | f3b6e15321 | ||
|   | 6591d5da63 | ||
|   | c974dc9400 | ||
|   | 1671d49d44 | ||
|   | 6b248ef140 | ||
|   | 735681d294 | ||
|   | a9085c65c5 | ||
|   | e312425b1c | ||
|   | 13fe064f6e | ||
|   | 958f98d7e5 | ||
|   | dfad3c4d8e | ||
|   | 37267f3f04 | ||
|   | b34538d991 | ||
|   | fc97bd1315 | ||
|   | dbf3721ec2 | ||
|   | 59afbe09b1 | ||
|   | bfeaa1b119 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | e1c3124698 | ||
|   | f2e22e103b | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | dda94f013e | ||
|   | caf00e7ead | ||
|   | f214440d2e | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 240c9ac511 | ||
|   | a2c9bc346a | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | b1cbb1c73a | ||
|   | 434b1e3245 | ||
|   | 497fdcaf4e | ||
|   | 52b95f2b62 | ||
|   | 83391af866 | ||
|   | 0a1786f39b | ||
|   | 3b069ac034 | ||
|   | 07882b918b | ||
|   | cc5ba71f06 | ||
|   | f16b8fbe2a | ||
|   | fe54d99356 | ||
|   | a49efb07ea | ||
|   | e4fd008441 | ||
|   | de12023311 | ||
|   | 60ebdc0ad6 | ||
|   | cbd9823ad6 | ||
|   | ce76303a32 | ||
|   | 246f17c6c8 | ||
|   | 4313635b01 | ||
|   | 7ca2bd0666 | ||
|   | 4c6075e962 | ||
|   | 8d48e99487 | ||
|   | 454a2d9e9e | ||
|   | b8c713d4b9 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2a8cb87232 | ||
|   | e5673e4817 | ||
|   | ed48392450 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 48d6bedf1c | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ec2dc6cd80 | ||
|   | 51e6eed72a | ||
|   | 422bffe1a6 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 31351c5f5c | ||
|   | c30cf2e0cd | ||
|   | e97cfb9b5e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 42100588d5 | ||
|   | bc2facc87f | ||
|   | b4c6c4b61e | ||
|   | 4e082f997c | ||
|   | 1512599f4f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6c8f0b54ad | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 419ee9d6e7 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 34b649aa01 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a9982abde8 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | a94cc62207 | ||
|   | fc1b3e674b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 603ad6c817 | ||
|   | 9c9a0e4496 | ||
|   | 7c33785c07 | ||
|   | 9c32d931bc | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | d7b2e002ce | ||
|   | 51e70f0a20 | ||
|   | bd257925bd | ||
|   | 15b1b83c66 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b06c0a0eba | ||
|   | e9746aa0e3 | ||
|   | bfaab21589 | ||
|   | 3849569bd1 | ||
|   | c40a7751b9 | ||
|   | f39463ff4e | ||
|   | 2ada8ec681 | ||
|   | bdbf1b57ce | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 4c6fdbb21f | ||
|   | 889c4378a9 | ||
|   | 06dd039083 | ||
|   | 00acbd4f1d | ||
|   | 28a1b9d1ac | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 4f8a931a61 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 716ebfe08a | ||
|   | e1760db85c | ||
|   | 3cbb5239fb | ||
|   | aed629269d | ||
|   | 59bf25edb1 | ||
|   | eb07876657 | ||
|   | ae3ac2b719 | ||
|   | 5b4b316bbc | ||
|   | 1583783a0b | ||
|   | bd5f05ff2b | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 9be6b28141 | ||
|   | 3de8c9073d | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | e90a2aa4eb | ||
|   | edef851c89 | ||
|   | 7b37e037e4 | ||
|   | e092627da6 | ||
|   | 0a03ca94c7 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 86dec8f344 | ||
|   | 0b079f57ed | ||
|   | ba80790826 | ||
|   | 2f8b2944f1 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 46ddba4579 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 1fe166e014 | ||
|   | 71bc483eec | ||
|   | 55917fcabe | ||
|   | fae4116504 | ||
|   | 83db0355f3 | ||
|   | 6647b32ac0 | ||
|   | 3365fc92fd | ||
|   | 1a6f32534c | ||
|   | ce5d4e9c92 | ||
|   | 0ab85b5122 | ||
|   | 2c9e690dfb | ||
|   | 344cc70cd5 | ||
|   | 73f0f1212d | ||
|   | a61f5ac64c | ||
|   | 6a5be992c0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | bcb0ae1ee5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 6bc385b49b | ||
|   | c6b3fd8bcb | ||
|   | 7cabb5bf98 | ||
|   | 011056a9d5 | ||
|   | 384a197c6f | ||
|   | 0542758641 | ||
|   | 848148afd5 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 42226adda0 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c2412c0955 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c58b68e597 | ||
|   | 1b4a52aaa1 | ||
|   | 86adc377a0 | ||
|   | c83b0bfca6 | ||
|   | 1b3a91f63f | ||
|   | 924a13f724 | ||
|   | 7236b8b682 | ||
|   | a3b85c64ca | ||
|   | 278ef3a364 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 0cefdc8475 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 915584551c | ||
|   | 855c9aa28d | ||
|   | e277a8e1ea | ||
|   | 6d8c507169 | ||
|   | 00050f7c7b | ||
|   | 5a278381e3 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | af95963c5f | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c724436fc4 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e288824963 | ||
|   | 312bb743b9 | ||
|   | 7436a97684 | ||
|   | cbaceb95af | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 2abf38f98e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b40e1f7d01 | ||
|   | 15d4ac8ba2 | ||
|   | fcb7349e8e | ||
|   | 648cfd9d05 | ||
|   | c3df7d3439 | ||
|   | a5cd545a1b | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 96227f785a | ||
|   | 7f98c4b794 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | d914a82dea | ||
|   | 1a72efa3c9 | ||
|   | aa6c58103f | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | ba6976bdbb | ||
|   | 7a1be6bff2 | ||
|   | c3bd587cbf | ||
|   | a57a3dbb30 | ||
|   | f52ebc7bf0 | ||
|   | abf910fd93 | ||
|   | 1b0aa193bd | ||
|   | eac12fe031 | ||
|   | 62f46b706e | ||
|   | df5af5c8ad | ||
|   | 67a97ffc4d | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 45b39f36d6 | ||
|   | a339853bc5 | ||
|   | 6c2e06d40b | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 2d6a4d108f | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 703db0f01f | ||
|   | 8b5d6bf3db | ||
|   | 2d93e95848 | ||
|   | ab8c75958d | ||
|   | 9db3923d35 | ||
|   | e2860ed36d | ||
|   | 82a5680217 | ||
|   | 588fd0207d | ||
|   | 6dea158de9 | ||
|   | d956269d5f | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | f269919410 | ||
|   | 5f00066dff | ||
|   | 705f542129 | ||
|   | f036292b72 | ||
|   | f8a43d5dab | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 53c106d448 | ||
|   | 0f37186c88 | ||
|   | 0fb55f3ae8 | ||
|   | 78822f6121 | ||
|   | 2ee1d7540e | ||
|   | 43b2527275 | ||
|   | 248b573c03 | ||
|   | b9f7428f2f | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | e35dad81d9 | ||
|   | 7d40eb2e24 | ||
|   | d2783a3fbe | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 8ad794e189 | ||
|   | 358db10fe3 | ||
|   | 556f47fe12 | ||
|   | 0d5a2b4382 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | c9bc9acd1a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | e9e209d290 | ||
|   | 2e593a0022 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 3526a4cf23 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f4791cac2d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | fdafd4eefb | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 87a8847a8d | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7c31c79bbc | ||
|   | bf2a9b02c6 | ||
|   | 348858780c | ||
|   | eb481ac1c0 | ||
|   | 9a2d7a64ac | ||
|   | 32a7f9cd5a | ||
|   | 92431b2f4b | ||
|   | b4b2a92225 | ||
|   | fd45e81a83 | ||
|   | 97d59dce9c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | f3479d982c | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | b3ba673d9a | ||
|   | 68b7427640 | ||
|   | 9c68100dc0 | ||
|   | 6e694ad9ff | ||
|   | 5db511afdf | ||
|   | a8de26f88a | ||
|   | 7a07f1e81d | ||
|   | 92524ae97a | ||
|   | 1c89f6da24 | ||
|   | d1a3e3b859 | ||
|   | 79ae594d54 | ||
|   | f753f6dc46 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 97fe5c4176 | ||
|   | 1f5086164b | ||
|   | e60bd3a132 | ||
|   | b4047e73bb | ||
|   | 4263d2196c | ||
|   | ac780134fb | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 5d6cfa7349 | ||
|   | 3105317137 | ||
|   | 1d9482acc3 | ||
| ![github-actions[bot]](/assets/img/avatar_default.png)  | 1456169d7f | ||
|   | 22a6fe5e10 | ||
|   | caa3c13edd | ||
|   | 24e863b298 | ||
|   | 0c9d615f56 | ||
|   | 90561857e8 | ||
|   | fc68f55d1a | ||
|   | 6a8ec182fa | ||
|   | 69541546ea | 
| @@ -10,10 +10,8 @@ component_management: | ||||
|       paths: | ||||
|         - src-ui/** | ||||
| # https://docs.codecov.com/docs/pull-request-comments | ||||
| # codecov will only comment if coverage changes | ||||
| comment: | ||||
|   layout: "header, diff, components, flags, files" | ||||
|   require_changes: true | ||||
|   # https://docs.codecov.com/docs/javascript-bundle-analysis | ||||
|   require_bundle_changes: true | ||||
|   bundle_change_threshold: "50Kb" | ||||
|   | ||||
| @@ -1,3 +0,0 @@ | ||||
| [codespell] | ||||
| write-changes = True | ||||
| ignore-words-list = criterias,afterall,valeu,ureue,equest,ure,assertIn | ||||
| @@ -83,7 +83,8 @@ RUN set -eux \ | ||||
|     && apt-get update \ | ||||
|     && apt-get install --yes --quiet ${PYTHON_PACKAGES} | ||||
|  | ||||
| COPY --from=ghcr.io/astral-sh/uv:0.6 /uv /bin/uv | ||||
| COPY --from=ghcr.io/astral-sh/uv:0.7.8 /uv /bin/uv | ||||
|  | ||||
|  | ||||
| RUN set -eux \ | ||||
|   && echo "Installing pre-built updates" \ | ||||
| @@ -128,7 +129,6 @@ RUN set -eux \ | ||||
|   && echo "Configuring ImageMagick" \ | ||||
|     && mv paperless-policy.xml /etc/ImageMagick-6/policy.xml | ||||
|  | ||||
| COPY --from=ghcr.io/astral-sh/uv:0.6 /uv /bin/uv | ||||
|  | ||||
| # Packages needed only for building a few quick Python | ||||
| # dependencies | ||||
|   | ||||
| @@ -47,39 +47,19 @@ To start the DevContainer: | ||||
|  | ||||
| 1. Open VSCode. | ||||
| 2. Open the project folder. | ||||
| 3. Open the command palette: | ||||
|    - **Windows/Linux**: `Ctrl+Shift+P` | ||||
|    - **Mac**: `Cmd+Shift+P` | ||||
| 4. Type and select `Dev Containers: Rebuild and Reopen in Container`. | ||||
| 3. Open the command palette and choose `Dev Containers: Rebuild and Reopen in Container`. | ||||
|  | ||||
| VSCode will build and start the DevContainer environment. | ||||
|  | ||||
| ### Step 2: Initial Setup | ||||
|  | ||||
| Once the DevContainer is up and running, perform the following steps: | ||||
| Once the DevContainer is up and running, run the `Project Setup: Run all Init Tasks` task to initialize the project. | ||||
|  | ||||
| 1. **Compile Frontend Assets**: | ||||
| Alternatively, the Project Setup can be done with individual tasks: | ||||
|  | ||||
|    - Open the command palette: | ||||
|      - **Windows/Linux**: `Ctrl+Shift+P` | ||||
|      - **Mac**: `Cmd+Shift+P` | ||||
|    - Select `Tasks: Run Task`. | ||||
|    - Choose `Frontend Compile`. | ||||
|  | ||||
| 2. **Run Database Migrations**: | ||||
|  | ||||
|    - Open the command palette: | ||||
|      - **Windows/Linux**: `Ctrl+Shift+P` | ||||
|      - **Mac**: `Cmd+Shift+P` | ||||
|    - Select `Tasks: Run Task`. | ||||
|    - Choose `Migrate Database`. | ||||
|  | ||||
| 3. **Create Superuser**: | ||||
|    - Open the command palette: | ||||
|      - **Windows/Linux**: `Ctrl+Shift+P` | ||||
|      - **Mac**: `Cmd+Shift+P` | ||||
|    - Select `Tasks: Run Task`. | ||||
|    - Choose `Create Superuser`. | ||||
| 1. **Compile Frontend Assets**: `Maintenance: Compile frontend for production`. | ||||
| 2. **Run Database Migrations**: `Maintenance: manage.py migrate`. | ||||
| 3. **Create Superuser**: `Maintenance: manage.py createsuperuser`. | ||||
|  | ||||
| ### Debugging and Running Services | ||||
|  | ||||
| @@ -95,11 +75,8 @@ You can start and debug backend services either as debugging sessions via `launc | ||||
|  | ||||
| #### Using Tasks | ||||
|  | ||||
| 1. Open the command palette: | ||||
|    - **Windows/Linux**: `Ctrl+Shift+P` | ||||
|    - **Mac**: `Cmd+Shift+P` | ||||
| 2. Select `Tasks: Run Task`. | ||||
| 3. Choose the desired task: | ||||
| 1. Open the command palette and select `Tasks: Run Task`. | ||||
| 2. Choose the desired task: | ||||
|    - `Runserver` | ||||
|    - `Document Consumer` | ||||
|    - `Celery` | ||||
|   | ||||
| @@ -3,7 +3,7 @@ | ||||
|     "dockerComposeFile": "docker-compose.devcontainer.sqlite-tika.yml", | ||||
|     "service": "paperless-development", | ||||
|     "workspaceFolder": "/usr/src/paperless/paperless-ngx", | ||||
|     "postCreateCommand": "/bin/bash -c 'uv sync --group dev && uv run pre-commit install'", | ||||
|     "postCreateCommand": "/bin/bash -c 'rm -rf .venv/.*  && uv sync --group dev && uv run pre-commit install'", | ||||
|     "customizations": { | ||||
|         "vscode": { | ||||
|           "extensions": [ | ||||
|   | ||||
| @@ -20,20 +20,17 @@ | ||||
| # | ||||
| # This file is intended only to be used through VSCOde devcontainers. See README.md | ||||
| # in the folder .devcontainer. | ||||
|  | ||||
|  | ||||
| services: | ||||
|   broker: | ||||
|     image: docker.io/library/redis:7 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - ./redisdata:/data | ||||
|  | ||||
|   # No ports need to be exposed; the VSCode DevContainer plugin manages them. | ||||
|   paperless-development: | ||||
|     image: paperless-ngx | ||||
|     build: | ||||
|       context: ../    # Dockerfile cannot access files from parent directories if context is not set. | ||||
|       context: ../ # Dockerfile cannot access files from parent directories if context is not set. | ||||
|       dockerfile: ./.devcontainer/Dockerfile | ||||
|     restart: unless-stopped | ||||
|     depends_on: | ||||
| @@ -52,7 +49,6 @@ services: | ||||
|       - ./data:/usr/src/paperless/paperless-ngx/data | ||||
|       - ./media:/usr/src/paperless/paperless-ngx/media | ||||
|       - ./consume:/usr/src/paperless/paperless-ngx/consume | ||||
|       - ~/.gitconfig:/usr/src/paperless/.gitconfig:ro | ||||
|     environment: | ||||
|       PAPERLESS_REDIS: redis://broker:6379 | ||||
|       PAPERLESS_TIKA_ENABLED: 1 | ||||
| @@ -60,25 +56,20 @@ services: | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
|       PAPERLESS_STATICDIR: ./src/documents/static | ||||
|       PAPERLESS_DEBUG: true | ||||
|  | ||||
|     # Overrides default command so things don't shut down after the process ends. | ||||
|     command: /bin/sh -c "chown -R paperless:paperless /usr/src/paperless/paperless-ngx/src/documents/static/frontend && chown -R paperless:paperless /usr/src/paperless/paperless-ngx/.ruff_cache && while sleep 1000; do :; done" | ||||
|  | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:8.17 | ||||
|     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. | ||||
|     command: | ||||
|       - "gotenberg" | ||||
|       - "--chromium-disable-javascript=true" | ||||
|       - "--chromium-allow-list=file:///tmp/.*" | ||||
|  | ||||
|   tika: | ||||
|     image: docker.io/apache/tika:latest | ||||
|     restart: unless-stopped | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -1,11 +1,10 @@ | ||||
| { | ||||
|     "python.testing.pytestArgs": [ | ||||
|         "src" | ||||
|     ], | ||||
|     "python.testing.pytestArgs": [], | ||||
|     "python.testing.unittestEnabled": false, | ||||
|     "python.testing.pytestEnabled": true, | ||||
|     "files.watcherExclude": { | ||||
|         "**/.venv/**": true, | ||||
|         "**/pytest_cache/**": true | ||||
|     } | ||||
|     }, | ||||
|     "python.testing.cwd": "${workspaceFolder}/src" | ||||
| } | ||||
|   | ||||
| @@ -156,7 +156,7 @@ | ||||
| 			"label": "Maintenance: recreate .venv", | ||||
| 			"description": "Recreate the python virtual environment and install python dependencies", | ||||
| 			"type": "shell", | ||||
| 			"command": "rm -R -v .venv/* || uv install --dev", | ||||
| 			"command": "rm -rf .venv && uv venv && uv sync --dev", | ||||
| 			"group": "none", | ||||
| 			"presentation": { | ||||
| 				"echo": true, | ||||
|   | ||||
							
								
								
									
										55
									
								
								.github/DISCUSSION_TEMPLATE/support.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								.github/DISCUSSION_TEMPLATE/support.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| title: "[Support] " | ||||
| body: | ||||
|   - type: textarea | ||||
|     id: description | ||||
|     attributes: | ||||
|       label: What's your question or issue? | ||||
|       description: Provide a clear and concise description of what you're trying to do, and what's going wrong. | ||||
|       placeholder: | | ||||
|         I'm trying to... | ||||
|  | ||||
|         [Include screenshots if helpful] | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: textarea | ||||
|     id: steps | ||||
|     attributes: | ||||
|       label: What have you tried? | ||||
|       description: Describe any steps you've already taken to troubleshoot or solve the issue. | ||||
|       placeholder: | | ||||
|         - I checked the logs and saw... | ||||
|         - I followed the install guide and tried... | ||||
|   - type: input | ||||
|     id: version | ||||
|     attributes: | ||||
|       label: Paperless-ngx version | ||||
|       placeholder: e.g. 1.14.0 | ||||
|     validations: | ||||
|       required: true | ||||
|   - type: input | ||||
|     id: host-os | ||||
|     attributes: | ||||
|       label: Host OS | ||||
|       description: Include architecture if relevant. | ||||
|       placeholder: e.g. Ubuntu 22.04 / Raspberry Pi arm64 | ||||
|   - type: dropdown | ||||
|     id: install-method | ||||
|     attributes: | ||||
|       label: Installation method | ||||
|       options: | ||||
|         - Docker - official image | ||||
|         - Docker - linuxserver.io image | ||||
|         - Bare metal | ||||
|         - Other (please describe above) | ||||
|   - type: textarea | ||||
|     id: system-status | ||||
|     attributes: | ||||
|       label: System status | ||||
|       description: If available, copy & paste the system status output from Settings > System Status > Copy | ||||
|       render: json | ||||
|   - type: textarea | ||||
|     id: logs | ||||
|     attributes: | ||||
|       label: Relevant logs or output | ||||
|       description: If you have logs, errors that might help, paste it here. | ||||
|       render: bash | ||||
							
								
								
									
										29
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										29
									
								
								.github/dependabot.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,11 +1,9 @@ | ||||
| # Please see the documentation for all configuration options: | ||||
| # https://docs.github.com/github/administering-a-repository/configuration-options-for-dependency-updates | ||||
|  | ||||
| version: 2 | ||||
| # Required for uv support for now | ||||
| enable-beta-ecosystems: true | ||||
| updates: | ||||
|  | ||||
|   # Enable version updates for pnpm | ||||
|   - package-ecosystem: "npm" | ||||
|     target-branch: "dev" | ||||
| @@ -17,9 +15,6 @@ updates: | ||||
|     labels: | ||||
|       - "frontend" | ||||
|       - "dependencies" | ||||
|     # Add reviewers | ||||
|     reviewers: | ||||
|       - "paperless-ngx/frontend" | ||||
|     groups: | ||||
|       frontend-angular-dependencies: | ||||
|         patterns: | ||||
| @@ -35,7 +30,6 @@ updates: | ||||
|         patterns: | ||||
|           - "@typescript-eslint*" | ||||
|           - "eslint" | ||||
|  | ||||
|   # Enable version updates for Python | ||||
|   - package-ecosystem: "uv" | ||||
|     target-branch: "dev" | ||||
| @@ -46,9 +40,6 @@ updates: | ||||
|     labels: | ||||
|       - "backend" | ||||
|       - "dependencies" | ||||
|     # Add reviewers | ||||
|     reviewers: | ||||
|       - "paperless-ngx/backend" | ||||
|     groups: | ||||
|       development: | ||||
|         patterns: | ||||
| @@ -59,6 +50,7 @@ updates: | ||||
|       django: | ||||
|         patterns: | ||||
|           - "*django*" | ||||
|           - "drf-*" | ||||
|       major-versions: | ||||
|         update-types: | ||||
|           - "major" | ||||
| @@ -66,11 +58,13 @@ updates: | ||||
|         update-types: | ||||
|           - "minor" | ||||
|           - "patch" | ||||
|         exclude-patterns: | ||||
|           - "*django*" | ||||
|           - "drf-*" | ||||
|       pre-built: | ||||
|         patterns: | ||||
|           - psycopg* | ||||
|           - zxing-cpp | ||||
|  | ||||
|   # Enable updates for GitHub Actions | ||||
|   - package-ecosystem: "github-actions" | ||||
|     target-branch: "dev" | ||||
| @@ -81,41 +75,32 @@ updates: | ||||
|     labels: | ||||
|       - "ci-cd" | ||||
|       - "dependencies" | ||||
|     # Add reviewers | ||||
|     reviewers: | ||||
|       - "paperless-ngx/ci-cd" | ||||
|     groups: | ||||
|       actions: | ||||
|         update-types: | ||||
|           - "major" | ||||
|           - "minor" | ||||
|           - "patch" | ||||
|  | ||||
|   # Update Dockerfile in root directory | ||||
|   - package-ecosystem: "docker" | ||||
|     directory: "/" | ||||
|     directories: | ||||
|       - "/" | ||||
|       - "/.devcontainer/" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     open-pull-requests-limit: 5 | ||||
|     reviewers: | ||||
|       - "paperless-ngx/ci-cd" | ||||
|     labels: | ||||
|       - "ci-cd" | ||||
|       - "dependencies" | ||||
|     commit-message: | ||||
|       prefix: "docker" | ||||
|       include: "scope" | ||||
|  | ||||
|   # Update Docker Compose files in docker/compose directory | ||||
|   - package-ecosystem: "docker-compose" | ||||
|     directory: "/docker/compose/" | ||||
|     schedule: | ||||
|       interval: "weekly" | ||||
|     open-pull-requests-limit: 5 | ||||
|     reviewers: | ||||
|       - "paperless-ngx/ci-cd" | ||||
|     labels: | ||||
|       - "ci-cd" | ||||
|       - "dependencies" | ||||
|     commit-message: | ||||
|       prefix: "docker-compose" | ||||
|   | ||||
							
								
								
									
										26
									
								
								.github/labeler.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								.github/labeler.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| backend: | ||||
|   - changed-files: | ||||
|       - any-glob-to-any-file: | ||||
|           - 'src/**' | ||||
|           - 'pyproject.toml' | ||||
|           - 'uv.lock' | ||||
|           - 'requirements.txt' | ||||
| frontend: | ||||
|   - changed-files: | ||||
|       - any-glob-to-any-file: | ||||
|           - 'src-ui/**' | ||||
| documentation: | ||||
|   - changed-files: | ||||
|       - any-glob-to-any-file: | ||||
|           - 'docs/**' | ||||
| ci-cd: | ||||
|   - changed-files: | ||||
|       - any-glob-to-any-file: | ||||
|           - '.github/**' | ||||
| # pr types | ||||
| bug: | ||||
|   - head-branch: | ||||
|       - ['^fix'] | ||||
| enhancement: | ||||
|   - head-branch: | ||||
|       - ['^feature'] | ||||
							
								
								
									
										14
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										14
									
								
								.github/release-drafter.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,15 +1,3 @@ | ||||
| autolabeler: | ||||
|   - label: "bug" | ||||
|     branch: | ||||
|       - '/^fix/' | ||||
|     title: | ||||
|       - "/^fix/i" | ||||
|       - "/^Bugfix/i" | ||||
|   - label: "enhancement" | ||||
|     branch: | ||||
|       - '/^feature/' | ||||
|     title: | ||||
|       - "/^feature/i" | ||||
| categories: | ||||
|   - title: 'Breaking Changes' | ||||
|     labels: | ||||
| @@ -17,7 +5,7 @@ categories: | ||||
|   - title: 'Notable Changes' | ||||
|     labels: | ||||
|       - 'notable' | ||||
|   - title: 'Features' | ||||
|   - title: 'Features / Enhancements' | ||||
|     labels: | ||||
|       - 'enhancement' | ||||
|   - title: 'Bug Fixes' | ||||
|   | ||||
							
								
								
									
										423
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										423
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,4 @@ | ||||
| name: ci | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     tags: | ||||
| @@ -12,72 +11,58 @@ on: | ||||
|   pull_request: | ||||
|     branches-ignore: | ||||
|       - 'translations**' | ||||
|  | ||||
| env: | ||||
|   DEFAULT_UV_VERSION: "0.6.x" | ||||
|   DEFAULT_UV_VERSION: "0.8.x" | ||||
|   # This is the default version of Python to use in most steps which aren't specific | ||||
|   DEFAULT_PYTHON_VERSION: "3.11" | ||||
|  | ||||
|   NLTK_DATA: "/usr/share/nltk_data" | ||||
| 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 | ||||
|  | ||||
|     if: github.event_name == 'push' || github.event.pull_request.head.repo.full_name != github.repository | ||||
|     name: Linting Checks | ||||
|     runs-on: ubuntu-24.04 | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout repository | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Install python | ||||
|       - name: Checkout repository | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Install python | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|       - | ||||
|         name: Check files | ||||
|       - name: Check files | ||||
|         uses: pre-commit/action@v3.0.1 | ||||
|  | ||||
|   documentation: | ||||
|     name: "Build & Deploy Documentation" | ||||
|     runs-on: ubuntu-24.04 | ||||
|     needs: | ||||
|       - pre-commit | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Set up Python | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Set up Python | ||||
|         id: setup-python | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|       - | ||||
|         name: Install uv | ||||
|         uses: astral-sh/setup-uv@v5 | ||||
|       - name: Install uv | ||||
|         uses: astral-sh/setup-uv@v6 | ||||
|         with: | ||||
|           version: ${{ env.DEFAULT_UV_VERSION }} | ||||
|           enable-cache: true | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|       - | ||||
|         name: Install Python dependencies | ||||
|       - name: Install Python dependencies | ||||
|         run: | | ||||
|           uv sync --python ${{ steps.setup-python.outputs.python-version }} --dev --frozen | ||||
|       - | ||||
|         name: Make documentation | ||||
|       - name: Make documentation | ||||
|         run: | | ||||
|           uv run \ | ||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ | ||||
|             --dev \ | ||||
|             --frozen \ | ||||
|             mkdocs build --config-file ./mkdocs.yml | ||||
|       - | ||||
|         name: Deploy documentation | ||||
|       - name: Deploy documentation | ||||
|         if: github.event_name == 'push' && github.ref == 'refs/heads/main' | ||||
|         run: | | ||||
|           echo "docs.paperless-ngx.com" > "${{ github.workspace }}/docs/CNAME" | ||||
| @@ -88,14 +73,12 @@ jobs: | ||||
|             --dev \ | ||||
|             --frozen \ | ||||
|             mkdocs gh-deploy --force --no-history | ||||
|       - | ||||
|         name: Upload artifact | ||||
|       - name: Upload artifact | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: documentation | ||||
|           path: site/ | ||||
|           retention-days: 7 | ||||
|  | ||||
|   tests-backend: | ||||
|     name: "Backend Tests (Python ${{ matrix.python-version }})" | ||||
|     runs-on: ubuntu-24.04 | ||||
| @@ -106,50 +89,44 @@ jobs: | ||||
|         python-version: ['3.10', '3.11', '3.12'] | ||||
|       fail-fast: false | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Start containers | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v5 | ||||
|       - 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 | ||||
|       - | ||||
|         name: Set up Python | ||||
|       - name: Set up Python | ||||
|         id: setup-python | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: "${{ matrix.python-version }}" | ||||
|       - | ||||
|         name: Install uv | ||||
|         uses: astral-sh/setup-uv@v5 | ||||
|       - name: Install uv | ||||
|         uses: astral-sh/setup-uv@v6 | ||||
|         with: | ||||
|           version: ${{ env.DEFAULT_UV_VERSION }} | ||||
|           enable-cache: true | ||||
|           python-version: ${{ steps.setup-python.outputs.python-version }} | ||||
|       - | ||||
|         name: Install system dependencies | ||||
|       - name: Install system dependencies | ||||
|         run: | | ||||
|           sudo apt-get update -qq | ||||
|           sudo apt-get install -qq --no-install-recommends unpaper tesseract-ocr imagemagick ghostscript libzbar0 poppler-utils | ||||
|       - | ||||
|         name: Configure ImageMagick | ||||
|       - name: Configure ImageMagick | ||||
|         run: | | ||||
|           sudo cp docker/rootfs/etc/ImageMagick-6/paperless-policy.xml /etc/ImageMagick-6/policy.xml | ||||
|       - | ||||
|         name: Install Python dependencies | ||||
|       - name: Install Python dependencies | ||||
|         run: | | ||||
|           uv sync \ | ||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ | ||||
|             --group testing \ | ||||
|             --frozen | ||||
|       - | ||||
|         name: List installed Python dependencies | ||||
|       - name: List installed Python dependencies | ||||
|         run: | | ||||
|           uv pip list | ||||
|       - | ||||
|         name: Tests | ||||
|       - name: Install or update NLTK dependencies | ||||
|         run: uv run python -m nltk.downloader punkt punkt_tab snowball_data stopwords -d ${{ env.NLTK_DATA }} | ||||
|       - name: Tests | ||||
|         env: | ||||
|           NLTK_DATA: ${{ env.NLTK_DATA }} | ||||
|           PAPERLESS_CI_TEST: 1 | ||||
|           # Enable paperless_mail testing against real server | ||||
|           PAPERLESS_MAIL_TEST_HOST: ${{ secrets.TEST_MAIL_HOST }} | ||||
| @@ -161,41 +138,48 @@ jobs: | ||||
|             --dev \ | ||||
|             --frozen \ | ||||
|             pytest | ||||
|       - | ||||
|         name: Upload backend test results to Codecov | ||||
|       - name: Upload backend test results to Codecov | ||||
|         if: always() | ||||
|         uses: codecov/test-results-action@v1 | ||||
|         with: | ||||
|           token: ${{ secrets.CODECOV_TOKEN }} | ||||
|           flags: backend-python-${{ matrix.python-version }} | ||||
|           files: junit.xml | ||||
|       - | ||||
|         name: Upload backend coverage to Codecov | ||||
|       - name: Upload backend coverage to Codecov | ||||
|         uses: codecov/codecov-action@v5 | ||||
|         with: | ||||
|           token: ${{ secrets.CODECOV_TOKEN }} | ||||
|           flags: backend-python-${{ matrix.python-version }} | ||||
|           files: coverage.xml | ||||
|       - | ||||
|         name: Stop containers | ||||
|       - name: Upload coverage artifacts | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         if: always() | ||||
|         with: | ||||
|           name: backend-coverage-${{ matrix.python-version }} | ||||
|           path: | | ||||
|             .coverage | ||||
|             coverage.xml | ||||
|             junit.xml | ||||
|           retention-days: 1 | ||||
|           include-hidden-files: true | ||||
|           if-no-files-found: error | ||||
|       - 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 | ||||
|  | ||||
|   install-frontend-dependencies: | ||||
|     name: "Install Frontend Dependencies" | ||||
|     runs-on: ubuntu-24.04 | ||||
|     needs: | ||||
|       - pre-commit | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
|       - name: Install pnpm | ||||
|         uses: pnpm/action-setup@v4 | ||||
|         with: | ||||
|           version: 10 | ||||
|       - | ||||
|         name: Use Node.js 20 | ||||
|       - name: Use Node.js 20 | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 20.x | ||||
| @@ -209,17 +193,10 @@ jobs: | ||||
|             ~/.pnpm-store | ||||
|             ~/.cache | ||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/pnpm-lock.yaml') }} | ||||
|       - | ||||
|         name: Install dependencies | ||||
|         if: steps.cache-frontend-deps.outputs.cache-hit != 'true' | ||||
|       - name: Install dependencies | ||||
|         run: cd src-ui && pnpm install | ||||
|       - | ||||
|         name: Install Playwright | ||||
|         if: steps.cache-frontend-deps.outputs.cache-hit != 'true' | ||||
|         run: cd src-ui && pnpm playwright install --with-deps | ||||
|  | ||||
|   tests-frontend: | ||||
|     name: "Frontend Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" | ||||
|     name: "Frontend Unit Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" | ||||
|     runs-on: ubuntu-24.04 | ||||
|     needs: | ||||
|       - install-frontend-dependencies | ||||
| @@ -230,13 +207,12 @@ jobs: | ||||
|         shard-index: [1, 2, 3, 4] | ||||
|         shard-count: [4] | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/checkout@v5 | ||||
|       - name: Install pnpm | ||||
|         uses: pnpm/action-setup@v4 | ||||
|         with: | ||||
|           version: 10 | ||||
|       - | ||||
|         name: Use Node.js 20 | ||||
|       - name: Use Node.js 20 | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 20.x | ||||
| @@ -252,52 +228,101 @@ jobs: | ||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/pnpm-lock.yaml') }} | ||||
|       - name: Re-link Angular cli | ||||
|         run: cd src-ui && pnpm link @angular/cli | ||||
|       - | ||||
|         name: Linting checks | ||||
|       - name: Linting checks | ||||
|         run: cd src-ui && pnpm run lint | ||||
|       - | ||||
|         name: Run Jest unit tests | ||||
|       - name: Run Jest unit tests | ||||
|         run: cd src-ui && pnpm run test --max-workers=2 --shard=${{ matrix.shard-index }}/${{ matrix.shard-count }} | ||||
|       - | ||||
|         name: Run Playwright e2e tests | ||||
|         run: cd src-ui && pnpm exec playwright test --shard ${{ matrix.shard-index }}/${{ matrix.shard-count }} | ||||
|       - | ||||
|         name: Upload frontend test results to Codecov | ||||
|       - name: Upload frontend test results to Codecov | ||||
|         uses: codecov/test-results-action@v1 | ||||
|         if: always() | ||||
|         with: | ||||
|           token: ${{ secrets.CODECOV_TOKEN }} | ||||
|           flags: frontend-node-${{ matrix.node-version }} | ||||
|           directory: src-ui/ | ||||
|       - | ||||
|         name: Upload frontend coverage to Codecov | ||||
|       - name: Upload frontend coverage to Codecov | ||||
|         uses: codecov/codecov-action@v5 | ||||
|         with: | ||||
|           token: ${{ secrets.CODECOV_TOKEN }} | ||||
|           flags: frontend-node-${{ matrix.node-version }} | ||||
|           directory: src-ui/coverage/ | ||||
|  | ||||
|   frontend-bundle-analysis: | ||||
|     name: "Frontend Bundle Analysis" | ||||
|       - name: Upload coverage artifacts | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         if: always() | ||||
|         with: | ||||
|           name: frontend-coverage-${{ matrix.shard-index }} | ||||
|           path: | | ||||
|             src-ui/coverage/lcov.info | ||||
|             src-ui/coverage/coverage-final.json | ||||
|             src-ui/junit.xml | ||||
|           retention-days: 1 | ||||
|           if-no-files-found: error | ||||
|   tests-frontend-e2e: | ||||
|     name: "Frontend E2E Tests (Node ${{ matrix.node-version }} - ${{ matrix.shard-index }}/${{ matrix.shard-count }})" | ||||
|     runs-on: ubuntu-24.04 | ||||
|     needs: | ||||
|       - tests-frontend | ||||
|       - install-frontend-dependencies | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         node-version: [20.x] | ||||
|         shard-index: [1, 2] | ||||
|         shard-count: [2] | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Install pnpm | ||||
|       - uses: actions/checkout@v5 | ||||
|       - name: Install pnpm | ||||
|         uses: pnpm/action-setup@v4 | ||||
|         with: | ||||
|           version: 10 | ||||
|       - | ||||
|         name: Use Node.js 20 | ||||
|       - name: Use Node.js 20 | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 20.x | ||||
|           cache: 'pnpm' | ||||
|           cache-dependency-path: 'src-ui/pnpm-lock.yaml' | ||||
|       - | ||||
|         name: Cache frontend dependencies | ||||
|       - name: Cache frontend dependencies | ||||
|         id: cache-frontend-deps | ||||
|         uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: | | ||||
|             ~/.pnpm-store | ||||
|             ~/.cache | ||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/pnpm-lock.yaml') }} | ||||
|       - name: Re-link Angular cli | ||||
|         run: cd src-ui && pnpm link @angular/cli | ||||
|       - name: Cache Playwright browsers | ||||
|         uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: ~/.cache/ms-playwright | ||||
|           key: ${{ runner.os }}-playwright-${{ hashFiles('src-ui/pnpm-lock.yaml') }} | ||||
|           restore-keys: | | ||||
|             ${{ runner.os }}-playwright- | ||||
|       - name: Install Playwright system dependencies | ||||
|         run: npx playwright install-deps | ||||
|       - name: Install dependencies | ||||
|         run: cd src-ui && pnpm install --no-frozen-lockfile | ||||
|       - name: Install Playwright | ||||
|         run: cd src-ui && pnpm exec playwright install | ||||
|       - name: Run Playwright e2e tests | ||||
|         run: cd src-ui && pnpm exec playwright test --shard ${{ matrix.shard-index }}/${{ matrix.shard-count }} | ||||
|   frontend-bundle-analysis: | ||||
|     name: "Frontend Bundle Analysis" | ||||
|     runs-on: ubuntu-24.04 | ||||
|     needs: | ||||
|       - tests-frontend | ||||
|       - tests-frontend-e2e | ||||
|     steps: | ||||
|       - uses: actions/checkout@v5 | ||||
|       - name: Install pnpm | ||||
|         uses: pnpm/action-setup@v4 | ||||
|         with: | ||||
|           version: 10 | ||||
|       - name: Use Node.js 20 | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 20.x | ||||
|           cache: 'pnpm' | ||||
|           cache-dependency-path: 'src-ui/pnpm-lock.yaml' | ||||
|       - name: Cache frontend dependencies | ||||
|         id: cache-frontend-deps | ||||
|         uses: actions/cache@v4 | ||||
|         with: | ||||
| @@ -305,28 +330,93 @@ jobs: | ||||
|             ~/.pnpm-store | ||||
|             ~/.cache | ||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/package-lock.json') }} | ||||
|       - | ||||
|         name: Re-link Angular cli | ||||
|       - name: Re-link Angular cli | ||||
|         run: cd src-ui && pnpm link @angular/cli | ||||
|       - | ||||
|         name: Build frontend and upload analysis | ||||
|       - name: Build frontend and upload analysis | ||||
|         env: | ||||
|           CODECOV_TOKEN: ${{ secrets.CODECOV_TOKEN }} | ||||
|         run: cd src-ui && pnpm run build --configuration=production | ||||
|  | ||||
|   sonarqube-analysis: | ||||
|     name: "SonarQube Analysis" | ||||
|     runs-on: ubuntu-24.04 | ||||
|     needs: | ||||
|       - tests-backend | ||||
|       - tests-frontend | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v5 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Download all backend coverage | ||||
|         uses: actions/download-artifact@v5.0.0 | ||||
|         with: | ||||
|           pattern: backend-coverage-* | ||||
|           path: ./coverage/ | ||||
|       - name: Download all frontend coverage | ||||
|         uses: actions/download-artifact@v5.0.0 | ||||
|         with: | ||||
|           pattern: frontend-coverage-* | ||||
|           path: ./coverage/ | ||||
|       - name: Set up Python | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|       - name: Install coverage tools | ||||
|         run: | | ||||
|           pip install coverage | ||||
|           npm install -g nyc | ||||
|       # Merge backend coverage from all Python versions | ||||
|       - name: Merge backend coverage | ||||
|         run: | | ||||
|           coverage combine coverage/backend-coverage-*/.coverage | ||||
|           coverage xml -o merged-backend-coverage.xml | ||||
|       # Merge frontend coverage from all shards | ||||
|       - name: Merge frontend coverage | ||||
|         run: | | ||||
|           # Find all coverage-final.json files from the shards, exit with error if none found | ||||
|           shopt -s nullglob | ||||
|           files=(coverage/frontend-coverage-*/coverage/coverage-final.json) | ||||
|           if [ ${#files[@]} -eq 0 ]; then | ||||
|             echo "No frontend coverage JSON found under coverage/" >&2 | ||||
|             exit 1 | ||||
|           fi | ||||
|           # Create .nyc_output directory and copy each shard's coverage JSON into it with a unique name | ||||
|           mkdir -p .nyc_output | ||||
|           for coverage_json in "${files[@]}"; do | ||||
|             shard=$(basename "$(dirname "$(dirname "$coverage_json")")") | ||||
|             cp "$coverage_json" ".nyc_output/${shard}.json" | ||||
|           done | ||||
|           npx nyc merge .nyc_output .nyc_output/out.json | ||||
|           npx nyc report --reporter=lcovonly --report-dir coverage | ||||
|       - name: Upload coverage artifacts | ||||
|         uses: actions/upload-artifact@v4.6.2 | ||||
|         with: | ||||
|           name: merged-coverage | ||||
|           path: | | ||||
|             merged-backend-coverage.xml | ||||
|             .nyc_output/* | ||||
|             coverage/lcov.info | ||||
|           retention-days: 7 | ||||
|           if-no-files-found: error | ||||
|           include-hidden-files: true | ||||
|       - name: SonarQube Analysis | ||||
|         uses: SonarSource/sonarqube-scan-action@v5 | ||||
|         env: | ||||
|           SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} | ||||
|   build-docker-image: | ||||
|     name: Build Docker image for ${{ github.ref_name }} | ||||
|     runs-on: ubuntu-24.04 | ||||
|     if: github.event_name == 'push' && (startsWith(github.ref, 'refs/heads/feature-') || startsWith(github.ref, 'refs/heads/fix-') || github.ref == 'refs/heads/dev' || github.ref == 'refs/heads/beta' || contains(github.ref, 'beta.rc') || startsWith(github.ref, 'refs/tags/v')) | ||||
|     if: github.event_name == 'push' && (startsWith(github.ref, 'refs/heads/feature-') || startsWith(github.ref, 'refs/heads/fix-') || github.ref == 'refs/heads/dev' || github.ref == 'refs/heads/beta' || contains(github.ref, 'beta.rc') || startsWith(github.ref, 'refs/tags/v') || startsWith(github.ref, 'refs/heads/l10n_')) | ||||
|     concurrency: | ||||
|       group: ${{ github.workflow }}-build-docker-image-${{ github.ref_name }} | ||||
|       cancel-in-progress: true | ||||
|     needs: | ||||
|       - tests-backend | ||||
|       - tests-frontend | ||||
|       - tests-frontend-e2e | ||||
|     steps: | ||||
|       - | ||||
|         name: Check pushing to Docker Hub | ||||
|       - name: Check pushing to Docker Hub | ||||
|         id: push-other-places | ||||
|         # Only push to Dockerhub from the main repo AND the ref is either: | ||||
|         #  main | ||||
| @@ -342,15 +432,13 @@ jobs: | ||||
|             echo "Not pushing to DockerHub" | ||||
|             echo "enable=false" >> $GITHUB_OUTPUT | ||||
|           fi | ||||
|       - | ||||
|         name: Set ghcr repository name | ||||
|       - name: Set ghcr repository name | ||||
|         id: set-ghcr-repository | ||||
|         run: | | ||||
|           ghcr_name=$(echo "${{ github.repository }}" | awk '{ print tolower($0) }') | ||||
|           echo "Name is ${ghcr_name}" | ||||
|           echo "ghcr-repository=${ghcr_name}" >> $GITHUB_OUTPUT | ||||
|       - | ||||
|         name: Gather Docker metadata | ||||
|       - name: Gather Docker metadata | ||||
|         id: docker-meta | ||||
|         uses: docker/metadata-action@v5 | ||||
|         with: | ||||
| @@ -365,37 +453,31 @@ jobs: | ||||
|             # For a tag x.y.z or vX.Y.Z, output an x.y.z and x.y image tag | ||||
|             type=semver,pattern={{version}} | ||||
|             type=semver,pattern={{major}}.{{minor}} | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v5 | ||||
|       # 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 | ||||
|       - name: Set up Docker Buildx | ||||
|         uses: docker/setup-buildx-action@v3 | ||||
|       - | ||||
|         name: Set up QEMU | ||||
|       - name: Set up QEMU | ||||
|         uses: docker/setup-qemu-action@v3 | ||||
|         with: | ||||
|           platforms: arm64 | ||||
|       - | ||||
|         name: Login to GitHub Container Registry | ||||
|       - 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 | ||||
|       - name: Login to Docker Hub | ||||
|         uses: docker/login-action@v3 | ||||
|         # Don't attempt to login if not pushing to Docker Hub | ||||
|         if: steps.push-other-places.outputs.enable == 'true' | ||||
|         with: | ||||
|           username: ${{ secrets.DOCKERHUB_USERNAME }} | ||||
|           password: ${{ secrets.DOCKERHUB_TOKEN }} | ||||
|       - | ||||
|         name: Login to Quay.io | ||||
|       - name: Login to Quay.io | ||||
|         uses: docker/login-action@v3 | ||||
|         # Don't attempt to login if not pushing to Quay.io | ||||
|         if: steps.push-other-places.outputs.enable == 'true' | ||||
| @@ -403,8 +485,7 @@ jobs: | ||||
|           registry: quay.io | ||||
|           username: ${{ secrets.QUAY_USERNAME }} | ||||
|           password: ${{ secrets.QUAY_ROBOT_TOKEN }} | ||||
|       - | ||||
|         name: Build and push | ||||
|       - name: Build and push | ||||
|         uses: docker/build-push-action@v6 | ||||
|         with: | ||||
|           context: . | ||||
| @@ -422,23 +503,19 @@ jobs: | ||||
|             type=registry,ref=ghcr.io/${{ steps.set-ghcr-repository.outputs.ghcr-repository }}/builder/cache/app:dev | ||||
|           cache-to: | | ||||
|             type=registry,mode=max,ref=ghcr.io/${{ steps.set-ghcr-repository.outputs.ghcr-repository }}/builder/cache/app:${{ github.ref_name }} | ||||
|       - | ||||
|         name: Inspect image | ||||
|       - name: Inspect image | ||||
|         run: | | ||||
|           docker buildx imagetools inspect ${{ fromJSON(steps.docker-meta.outputs.json).tags[0] }} | ||||
|       - | ||||
|         name: Export frontend artifact from docker | ||||
|       - name: Export frontend artifact from docker | ||||
|         run: | | ||||
|           docker create --name frontend-extract ${{ fromJSON(steps.docker-meta.outputs.json).tags[0] }} | ||||
|           docker cp frontend-extract:/usr/src/paperless/src/documents/static/frontend src/documents/static/frontend/ | ||||
|       - | ||||
|         name: Upload frontend artifact | ||||
|       - name: Upload frontend artifact | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: frontend-compiled | ||||
|           path: src/documents/static/frontend/ | ||||
|           retention-days: 7 | ||||
|  | ||||
|   build-release: | ||||
|     name: "Build Release" | ||||
|     needs: | ||||
| @@ -446,63 +523,52 @@ jobs: | ||||
|       - documentation | ||||
|     runs-on: ubuntu-24.04 | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - | ||||
|         name: Set up Python | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v5 | ||||
|       - name: Set up Python | ||||
|         id: setup-python | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|       - | ||||
|         name: Install uv | ||||
|         uses: astral-sh/setup-uv@v5 | ||||
|       - name: Install uv | ||||
|         uses: astral-sh/setup-uv@v6 | ||||
|         with: | ||||
|           version: ${{ env.DEFAULT_UV_VERSION }} | ||||
|           enable-cache: true | ||||
|           python-version: ${{ steps.setup-python.outputs.python-version }} | ||||
|       - | ||||
|         name: Install Python dependencies | ||||
|       - name: Install Python dependencies | ||||
|         run: | | ||||
|           uv sync --python ${{ steps.setup-python.outputs.python-version }} --dev --frozen | ||||
|       - | ||||
|         name: Install system dependencies | ||||
|       - name: Install system dependencies | ||||
|         run: | | ||||
|           sudo apt-get update -qq | ||||
|           sudo apt-get install -qq --no-install-recommends gettext liblept5 | ||||
|       - | ||||
|         name: Download frontend artifact | ||||
|         uses: actions/download-artifact@v4 | ||||
|       - name: Download frontend artifact | ||||
|         uses: actions/download-artifact@v5 | ||||
|         with: | ||||
|           name: frontend-compiled | ||||
|           path: src/documents/static/frontend/ | ||||
|       - | ||||
|         name: Download documentation artifact | ||||
|         uses: actions/download-artifact@v4 | ||||
|       - name: Download documentation artifact | ||||
|         uses: actions/download-artifact@v5 | ||||
|         with: | ||||
|           name: documentation | ||||
|           path: docs/_build/html/ | ||||
|       - | ||||
|         name: Generate requirements file | ||||
|       - name: Generate requirements file | ||||
|         run: | | ||||
|            uv export --quiet --no-dev --all-extras --format requirements-txt --output-file requirements.txt | ||||
|       - | ||||
|         name: Compile messages | ||||
|           uv export --quiet --no-dev --all-extras --format requirements-txt --output-file requirements.txt | ||||
|       - name: Compile messages | ||||
|         run: | | ||||
|           cd src/ | ||||
|           uv run \ | ||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ | ||||
|             manage.py compilemessages | ||||
|       - | ||||
|         name: Collect static files | ||||
|       - name: Collect static files | ||||
|         run: | | ||||
|           cd src/ | ||||
|           uv run \ | ||||
|             --python ${{ steps.setup-python.outputs.python-version }} \ | ||||
|             manage.py collectstatic --no-input | ||||
|       - | ||||
|         name: Move files | ||||
|       - name: Move files | ||||
|         run: | | ||||
|           echo "Making dist folders" | ||||
|           for directory in dist \ | ||||
| @@ -539,21 +605,18 @@ jobs: | ||||
|           cp --recursive docs/_build/html/ dist/paperless-ngx/docs | ||||
|  | ||||
|           mv --verbose static dist/paperless-ngx | ||||
|       - | ||||
|         name: Make release package | ||||
|       - name: Make release package | ||||
|         run: | | ||||
|           echo "Creating release archive" | ||||
|           cd dist | ||||
|           sudo chown -R 1000:1000 paperless-ngx/ | ||||
|           tar -cJf paperless-ngx.tar.xz paperless-ngx/ | ||||
|       - | ||||
|         name: Upload release artifact | ||||
|       - name: Upload release artifact | ||||
|         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-24.04 | ||||
| @@ -565,14 +628,12 @@ jobs: | ||||
|       - build-release | ||||
|     if: github.ref_type == 'tag' && (startsWith(github.ref_name, 'v') || contains(github.ref_name, '-beta.rc')) | ||||
|     steps: | ||||
|       - | ||||
|         name: Download release artifact | ||||
|         uses: actions/download-artifact@v4 | ||||
|       - name: Download release artifact | ||||
|         uses: actions/download-artifact@v5 | ||||
|         with: | ||||
|           name: release | ||||
|           path: ./ | ||||
|       - | ||||
|         name: Get version | ||||
|       - name: Get version | ||||
|         id: get_version | ||||
|         run: | | ||||
|           echo "version=${{ github.ref_name }}" >> $GITHUB_OUTPUT | ||||
| @@ -581,8 +642,7 @@ jobs: | ||||
|           else | ||||
|             echo "prerelease=false" >> $GITHUB_OUTPUT | ||||
|           fi | ||||
|       - | ||||
|         name: Create Release and Changelog | ||||
|       - name: Create Release and Changelog | ||||
|         id: create-release | ||||
|         uses: release-drafter/release-drafter@v6 | ||||
|         with: | ||||
| @@ -593,8 +653,7 @@ jobs: | ||||
|           publish: true # ensures release is not marked as draft | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|       - | ||||
|         name: Upload release archive | ||||
|       - name: Upload release archive | ||||
|         id: upload-release-asset | ||||
|         uses: shogo82148/actions-upload-release-asset@v1 | ||||
|         with: | ||||
| @@ -603,7 +662,6 @@ jobs: | ||||
|           asset_path: ./paperless-ngx.tar.xz | ||||
|           asset_name: paperless-ngx-${{ steps.get_version.outputs.version }}.tar.xz | ||||
|           asset_content_type: application/x-xz | ||||
|  | ||||
|   append-changelog: | ||||
|     name: "Append Changelog" | ||||
|     runs-on: ubuntu-24.04 | ||||
| @@ -611,26 +669,22 @@ jobs: | ||||
|       - publish-release | ||||
|     if: needs.publish-release.outputs.prerelease == 'false' | ||||
|     steps: | ||||
|       - | ||||
|         name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v5 | ||||
|         with: | ||||
|           ref: main | ||||
|       - | ||||
|         name: Set up Python | ||||
|       - name: Set up Python | ||||
|         id: setup-python | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|       - | ||||
|         name: Install uv | ||||
|         uses: astral-sh/setup-uv@v5 | ||||
|       - name: Install uv | ||||
|         uses: astral-sh/setup-uv@v6 | ||||
|         with: | ||||
|           version: ${{ env.DEFAULT_UV_VERSION }} | ||||
|           enable-cache: true | ||||
|           python-version: ${{ env.DEFAULT_PYTHON_VERSION }} | ||||
|       - | ||||
|         name: Append Changelog to docs | ||||
|       - name: Append Changelog to docs | ||||
|         id: append-Changelog | ||||
|         working-directory: docs | ||||
|         run: | | ||||
| @@ -652,8 +706,7 @@ jobs: | ||||
|           git config --global user.email "41898282+github-actions[bot]@users.noreply.github.com" | ||||
|           git commit -am "Changelog ${{ needs.publish-release.outputs.version }} - GHA" | ||||
|           git push origin ${{ needs.publish-release.outputs.version }}-changelog | ||||
|       - | ||||
|         name: Create Pull Request | ||||
|       - name: Create Pull Request | ||||
|         uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|   | ||||
							
								
								
									
										15
									
								
								.github/workflows/cleanup-tags.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										15
									
								
								.github/workflows/cleanup-tags.yml
									
									
									
									
										vendored
									
									
								
							| @@ -4,19 +4,15 @@ | ||||
| # Requires a PAT with the correct scope set in the secrets. | ||||
| # | ||||
| # This workflow will not trigger runs on forked repos. | ||||
|  | ||||
| name: Cleanup Image Tags | ||||
|  | ||||
| on: | ||||
|   delete: | ||||
|   push: | ||||
|     paths: | ||||
|       - ".github/workflows/cleanup-tags.yml" | ||||
|  | ||||
| concurrency: | ||||
|   group: registry-tags-cleanup | ||||
|   cancel-in-progress: false | ||||
|  | ||||
| jobs: | ||||
|   cleanup-images: | ||||
|     name: Cleanup Image Tags for ${{ matrix.primary-name }} | ||||
| @@ -30,10 +26,9 @@ jobs: | ||||
|       # Requires a personal access token with the OAuth scope delete:packages | ||||
|       TOKEN: ${{ secrets.GHA_CONTAINER_DELETE_TOKEN }} | ||||
|     steps: | ||||
|       - | ||||
|         name: Clean temporary images | ||||
|       - name: Clean temporary images | ||||
|         if: "${{ env.TOKEN != '' }}" | ||||
|         uses: stumpylog/image-cleaner-action/ephemeral@v0.10.0 | ||||
|         uses: stumpylog/image-cleaner-action/ephemeral@v0.11.0 | ||||
|         with: | ||||
|           token: "${{ env.TOKEN }}" | ||||
|           owner: "${{ github.repository_owner }}" | ||||
| @@ -43,7 +38,6 @@ jobs: | ||||
|           repo_name: "paperless-ngx" | ||||
|           match_regex: "(feature|fix)" | ||||
|           do_delete: "true" | ||||
|  | ||||
|   cleanup-untagged-images: | ||||
|     name: Cleanup Untagged Images Tags for ${{ matrix.primary-name }} | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
| @@ -58,10 +52,9 @@ jobs: | ||||
|       # Requires a personal access token with the OAuth scope delete:packages | ||||
|       TOKEN: ${{ secrets.GHA_CONTAINER_DELETE_TOKEN }} | ||||
|     steps: | ||||
|       - | ||||
|         name: Clean untagged images | ||||
|       - name: Clean untagged images | ||||
|         if: "${{ env.TOKEN != '' }}" | ||||
|         uses: stumpylog/image-cleaner-action/untagged@v0.10.0 | ||||
|         uses: stumpylog/image-cleaner-action/untagged@v0.11.0 | ||||
|         with: | ||||
|           token: "${{ env.TOKEN }}" | ||||
|           owner: "${{ github.repository_owner }}" | ||||
|   | ||||
							
								
								
									
										38
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										38
									
								
								.github/workflows/codeql-analysis.yml
									
									
									
									
										vendored
									
									
								
							| @@ -10,16 +10,14 @@ | ||||
| # supported CodeQL languages. | ||||
| # | ||||
| name: "CodeQL" | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: [ main, dev ] | ||||
|     branches: [main, dev] | ||||
|   pull_request: | ||||
|     # The branches below must be a subset of the branches above | ||||
|     branches: [ dev ] | ||||
|     branches: [dev] | ||||
|   schedule: | ||||
|     - cron: '28 13 * * 5' | ||||
|  | ||||
| jobs: | ||||
|   analyze: | ||||
|     name: Analyze | ||||
| @@ -28,27 +26,23 @@ jobs: | ||||
|       actions: read | ||||
|       contents: read | ||||
|       security-events: write | ||||
|  | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         language: [ 'javascript', 'python' ] | ||||
|         language: ['javascript', 'python'] | ||||
|         # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] | ||||
|         # Learn more about CodeQL language support at https://git.io/codeql-language-support | ||||
|  | ||||
|     steps: | ||||
|     - name: Checkout repository | ||||
|       uses: actions/checkout@v4 | ||||
|  | ||||
|     # Initializes the CodeQL tools for scanning. | ||||
|     - name: Initialize CodeQL | ||||
|       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. | ||||
|         # By default, queries listed here will override any specified in a config file. | ||||
|         # Prefix the list here with "+" to use these queries and those in the config file. | ||||
|         # queries: ./path/to/local/query, your-org/your-repo/queries@main | ||||
|  | ||||
|     - name: Perform CodeQL Analysis | ||||
|       uses: github/codeql-action/analyze@v3 | ||||
|       - name: Checkout repository | ||||
|         uses: actions/checkout@v5 | ||||
|       # Initializes the CodeQL tools for scanning. | ||||
|       - name: Initialize CodeQL | ||||
|         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. | ||||
|           # By default, queries listed here will override any specified in a config file. | ||||
|           # Prefix the list here with "+" to use these queries and those in the config file. | ||||
|           # queries: ./path/to/local/query, your-org/your-repo/queries@main | ||||
|       - name: Perform CodeQL Analysis | ||||
|         uses: github/codeql-action/analyze@v3 | ||||
|   | ||||
							
								
								
									
										41
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										41
									
								
								.github/workflows/crowdin.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,35 +1,30 @@ | ||||
| name: Crowdin Action | ||||
|  | ||||
| on: | ||||
|   workflow_dispatch: | ||||
|   schedule: | ||||
|     - cron: '2 */12 * * *' | ||||
|   push: | ||||
|     paths: [ | ||||
|       'src/locale/**', | ||||
|       'src-ui/messages.xlf', | ||||
|       'src-ui/src/locale/**' | ||||
|     ] | ||||
|     branches: [ dev ] | ||||
|  | ||||
|     paths: ['src/locale/**', 'src-ui/messages.xlf', 'src-ui/src/locale/**'] | ||||
|     branches: [dev] | ||||
| jobs: | ||||
|   synchronize-with-crowdin: | ||||
|     name: Crowdin Sync | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
|     runs-on: ubuntu-24.04 | ||||
|  | ||||
|     steps: | ||||
|     - name: Checkout | ||||
|       uses: actions/checkout@v4 | ||||
|     - name: crowdin action | ||||
|       uses: crowdin/github-action@v2 | ||||
|       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 }} | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v5 | ||||
|         with: | ||||
|           token: ${{ secrets.PNGX_BOT_PAT }} | ||||
|       - name: crowdin action | ||||
|         uses: crowdin/github-action@v2 | ||||
|         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 }} | ||||
|   | ||||
							
								
								
									
										112
									
								
								.github/workflows/pr-bot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								.github/workflows/pr-bot.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,112 @@ | ||||
| name: PR Bot | ||||
| on: | ||||
|   pull_request_target: | ||||
|     types: [opened] | ||||
| permissions: | ||||
|   contents: read | ||||
|   pull-requests: write | ||||
| jobs: | ||||
|   pr-bot: | ||||
|     name: Automated PR Bot | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Label PR by file path or branch name | ||||
|         # see .github/labeler.yml for the labeler config | ||||
|         uses: actions/labeler@v5 | ||||
|         with: | ||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|       - name: Label by size | ||||
|         uses: Gascon1/pr-size-labeler@v1.3.0 | ||||
|         with: | ||||
|           GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|           xs_label: 'small-change' | ||||
|           xs_diff: '9' | ||||
|           s_label: 'non-trivial' | ||||
|           s_diff: '99999' | ||||
|           fail_if_xl: 'false' | ||||
|           excluded_files: /\.lock$/ /\.txt$/ ^src-ui/pnpm-lock\.yaml$ ^src-ui/messages\.xlf$ ^src/locale/en_US/LC_MESSAGES/django\.po$ | ||||
|       - name: Label by PR title | ||||
|         uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|             const pr = context.payload.pull_request; | ||||
|             const title = pr.title.toLowerCase(); | ||||
|             const labels = []; | ||||
|  | ||||
|             if (/^(fix|bugfix)/i.test(title)) { | ||||
|               labels.push('bug'); | ||||
|             } else if (/^feature/i.test(title)) { | ||||
|               labels.push('enhancement'); | ||||
|             } else if (!/^(dependabot)/i.test(title) && !/^(chore)/i.test(title)) { | ||||
|               labels.push('enhancement'); // Default fallback | ||||
|             } | ||||
|  | ||||
|             if (labels.length) { | ||||
|               await github.rest.issues.addLabels({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: pr.number, | ||||
|                 labels, | ||||
|               }); | ||||
|               core.info(`Added labels based on title: ${labels.join(', ')}`); | ||||
|             } | ||||
|       - name: Label bot-generated PRs | ||||
|         if: ${{ contains(github.actor, 'dependabot') || contains(github.actor, 'crowdin-bot') }} | ||||
|         uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|             const pr = context.payload.pull_request; | ||||
|             const user = pr.user.login.toLowerCase(); | ||||
|             const labels = []; | ||||
|  | ||||
|             if (user.includes('dependabot')) { | ||||
|               labels.push('dependencies'); | ||||
|             } | ||||
|  | ||||
|             if (user.includes('crowdin-bot')) { | ||||
|               labels.push('translation', 'skip-changelog'); | ||||
|             } | ||||
|  | ||||
|             if (labels.length) { | ||||
|               await github.rest.issues.addLabels({ | ||||
|                 owner: context.repo.owner, | ||||
|                 repo: context.repo.repo, | ||||
|                 issue_number: pr.number, | ||||
|                 labels, | ||||
|               }); | ||||
|             } | ||||
|       - name: Welcome comment | ||||
|         if: ${{ !contains(github.actor, 'bot') }} | ||||
|         uses: actions/github-script@v7 | ||||
|         with: | ||||
|           script: | | ||||
|             const pr = context.payload.pull_request; | ||||
|             const user = pr.user.login; | ||||
|  | ||||
|             const { data: members } = await github.rest.orgs.listMembers({ | ||||
|               org: 'paperless-ngx', | ||||
|             }); | ||||
|  | ||||
|             const memberLogins = members.map(m => m.login.toLowerCase()); | ||||
|             if (memberLogins.includes(user.toLowerCase())) { | ||||
|               core.info('Skipping comment: user is org member'); | ||||
|               return; | ||||
|             } | ||||
|  | ||||
|             const body = | ||||
|                 "Hello @" + user + ",\n\n" + | ||||
|                 "Thank you very much for submitting this PR to us!\n\n" + | ||||
|                 "This is what will happen next:\n\n" + | ||||
|                 "1. CI tests will run against your PR to ensure quality and consistency.\n" + | ||||
|                 "2. Next, human contributors from paperless-ngx review your changes.\n" + | ||||
|                 "3. Please address any issues that come up during the review as soon as you are able to.\n" + | ||||
|                 "4. If accepted, your pull request will be merged into the `dev` branch and changes there will be tested further.\n" + | ||||
|                 "5. Eventually, changes from you and other contributors will be merged into `main` and a new release will be made.\n\n" + | ||||
|                 "You'll be hearing from us soon, and thank you again for contributing to our project."; | ||||
|  | ||||
|             await github.rest.issues.createComment({ | ||||
|               issue_number: pr.number, | ||||
|               owner: context.repo.owner, | ||||
|               repo: context.repo.repo, | ||||
|               body, | ||||
|             }); | ||||
							
								
								
									
										3
									
								
								.github/workflows/project-actions.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/workflows/project-actions.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,4 @@ | ||||
| name: Project Automations | ||||
|  | ||||
| on: | ||||
|   pull_request_target: #_target allows access to secrets | ||||
|     types: | ||||
| @@ -8,10 +7,8 @@ on: | ||||
|     branches: | ||||
|       - main | ||||
|       - dev | ||||
|  | ||||
| permissions: | ||||
|   contents: read | ||||
|  | ||||
| jobs: | ||||
|   pr_opened_or_reopened: | ||||
|     name: pr_opened_or_reopened | ||||
|   | ||||
							
								
								
									
										39
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										39
									
								
								.github/workflows/repo-maintenance.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,18 +1,14 @@ | ||||
| name: 'Repository Maintenance' | ||||
|  | ||||
| on: | ||||
|   schedule: | ||||
|     - cron: '0 3 * * *' | ||||
|   workflow_dispatch: | ||||
|  | ||||
| permissions: | ||||
|   issues: write | ||||
|   pull-requests: write | ||||
|   discussions: write | ||||
|  | ||||
| concurrency: | ||||
|   group: lock | ||||
|  | ||||
| jobs: | ||||
|   stale: | ||||
|     name: 'Stale' | ||||
| @@ -23,13 +19,19 @@ jobs: | ||||
|         with: | ||||
|           days-before-stale: 7 | ||||
|           days-before-close: 14 | ||||
|           any-of-labels: 'stale,cant-reproduce,not a bug' | ||||
|           any-of-issue-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. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||
|             This issue has been automatically marked as stale because it has not had recent activity. It will be closed if no further activity occurs. Thank you for your contributions. See our [contributing guidelines](https://github.com/paperless-ngx/paperless-ngx/blob/dev/CONTRIBUTING.md#automatic-repository-maintenance) for more details. | ||||
|  | ||||
|           days-before-pr-stale: 14 | ||||
|           days-before-pr-close: 7 | ||||
|           stale-pr-message: "" | ||||
|           stale-pr-label: stale | ||||
|           exempt-pr-labels: 'notable' | ||||
|           close-pr-message: > | ||||
|             This pull request has been automatically closed because it has not had recent activity. Thank you for your contributions. Please open a new pull request or discussion if you would like to continue working on this change. | ||||
|  | ||||
|   lock-threads: | ||||
|     name: 'Lock Old Threads' | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
| @@ -42,20 +44,14 @@ jobs: | ||||
|           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. | ||||
|             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. | ||||
|             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. | ||||
|             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' | ||||
|     if: github.repository_owner == 'paperless-ngx' | ||||
| @@ -245,6 +241,7 @@ jobs: | ||||
|                 ) { | ||||
|                   nodes { | ||||
|                     id, | ||||
|                     createdAt, | ||||
|                     number, | ||||
|                     updatedAt, | ||||
|                     upvoteCount, | ||||
|   | ||||
							
								
								
									
										69
									
								
								.github/workflows/translate-strings.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								.github/workflows/translate-strings.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| name: Generate Translation Strings | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - dev | ||||
| jobs: | ||||
|   generate-translate-strings: | ||||
|     name: Generate Translation Strings | ||||
|     runs-on: ubuntu-latest | ||||
|     permissions: | ||||
|       contents: write | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v5 | ||||
|         with: | ||||
|           token: ${{ secrets.PNGX_BOT_PAT }} | ||||
|           ref: ${{ github.head_ref }} | ||||
|       - name: Set up Python | ||||
|         id: setup-python | ||||
|         uses: actions/setup-python@v5 | ||||
|       - name: Install system dependencies | ||||
|         run: | | ||||
|           sudo apt-get update -qq | ||||
|           sudo apt-get install -qq --no-install-recommends gettext | ||||
|       - name: Install uv | ||||
|         uses: astral-sh/setup-uv@v6 | ||||
|         with: | ||||
|           enable-cache: true | ||||
|       - name: Install backend python dependencies | ||||
|         run: | | ||||
|           uv sync \ | ||||
|             --group dev \ | ||||
|             --frozen | ||||
|       - name: Generate backend translation strings | ||||
|         run: cd src/ && uv run manage.py makemessages -l en_US -i "samples*" | ||||
|       - name: Install pnpm | ||||
|         uses: pnpm/action-setup@v4 | ||||
|         with: | ||||
|           version: 10 | ||||
|       - name: Use Node.js 20 | ||||
|         uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version: 20.x | ||||
|           cache: 'pnpm' | ||||
|           cache-dependency-path: 'src-ui/pnpm-lock.yaml' | ||||
|       - name: Cache frontend dependencies | ||||
|         id: cache-frontend-deps | ||||
|         uses: actions/cache@v4 | ||||
|         with: | ||||
|           path: | | ||||
|             ~/.pnpm-store | ||||
|             ~/.cache | ||||
|           key: ${{ runner.os }}-frontenddeps-${{ hashFiles('src-ui/pnpm-lock.yaml') }} | ||||
|       - name: Install frontend dependencies | ||||
|         if: steps.cache-frontend-deps.outputs.cache-hit != 'true' | ||||
|         run: cd src-ui && pnpm install | ||||
|       - name: Re-link Angular cli | ||||
|         run: cd src-ui && pnpm link @angular/cli | ||||
|       - name: Generate frontend translation strings | ||||
|         run: | | ||||
|           cd src-ui | ||||
|           pnpm run ng extract-i18n | ||||
|       - name: Commit changes | ||||
|         uses: stefanzweifel/git-auto-commit-action@v6 | ||||
|         with: | ||||
|           file_pattern: 'src-ui/messages.xlf src/locale/en_US/LC_MESSAGES/django.po' | ||||
|           commit_message: "Auto translate strings" | ||||
|           commit_user_name: "GitHub Actions" | ||||
|           commit_author: "GitHub Actions <41898282+github-actions[bot]@users.noreply.github.com>" | ||||
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -107,3 +107,6 @@ celerybeat-schedule* | ||||
| /.devcontainer/data/ | ||||
| /.devcontainer/media/ | ||||
| /.devcontainer/redisdata/ | ||||
|  | ||||
| # ignore pnpm package store folder created when setting up the devcontainer | ||||
| .pnpm-store/ | ||||
|   | ||||
| @@ -1,11 +1,10 @@ | ||||
| # This file configures pre-commit hooks. | ||||
| # See https://pre-commit.com/ for general information | ||||
| # See https://pre-commit.com/hooks.html for a listing of possible hooks | ||||
|  | ||||
| repos: | ||||
|   # General hooks | ||||
|   - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|     rev: v5.0.0 | ||||
|     rev: v6.0.0 | ||||
|     hooks: | ||||
|       - id: check-docstring-first | ||||
|       - id: check-json | ||||
| @@ -19,7 +18,7 @@ repos: | ||||
|         exclude_types: | ||||
|           - svg | ||||
|           - pofile | ||||
|         exclude: "(^LICENSE$)" | ||||
|         exclude: "(^LICENSE$|^src/documents/static/bootstrap.min.css$)" | ||||
|       - id: mixed-line-ending | ||||
|         args: | ||||
|           - "--fix=lf" | ||||
| @@ -29,16 +28,16 @@ repos: | ||||
|       - id: check-case-conflict | ||||
|       - id: detect-private-key | ||||
|   - repo: https://github.com/codespell-project/codespell | ||||
|     rev: v2.4.0 | ||||
|     rev: v2.4.1 | ||||
|     hooks: | ||||
|       - id: codespell | ||||
|         exclude: "(^src-ui/src/locale/)|(^src-ui/pnpm-lock.yaml)|(^src-ui/e2e/)|(^src/paperless_mail/tests/samples/)" | ||||
|         additional_dependencies: [tomli] | ||||
|         exclude_types: | ||||
|           - pofile | ||||
|           - json | ||||
|   # See https://github.com/prettier/prettier/issues/15742 for the fork reason | ||||
|   - repo: https://github.com/rbubley/mirrors-prettier | ||||
|     rev: 'v3.3.3' | ||||
|     rev: 'v3.6.2' | ||||
|     hooks: | ||||
|       - id: prettier | ||||
|         types_or: | ||||
| @@ -50,17 +49,17 @@ repos: | ||||
|           - 'prettier-plugin-organize-imports@4.1.0' | ||||
|   # Python hooks | ||||
|   - repo: https://github.com/astral-sh/ruff-pre-commit | ||||
|     rev: v0.9.9 | ||||
|     rev: v0.13.0 | ||||
|     hooks: | ||||
|       - id: ruff | ||||
|       - id: ruff-check | ||||
|       - id: ruff-format | ||||
|   - repo: https://github.com/tox-dev/pyproject-fmt | ||||
|     rev: "v2.5.1" | ||||
|     rev: "v2.6.0" | ||||
|     hooks: | ||||
|       - id: pyproject-fmt | ||||
|   # Dockerfile hooks | ||||
|   - repo: https://github.com/AleksaC/hadolint-py | ||||
|     rev: v2.12.0.3 | ||||
|     rev: v2.12.1b3 | ||||
|     hooks: | ||||
|       - id: hadolint | ||||
|   # Shell script hooks | ||||
| @@ -73,6 +72,11 @@ repos: | ||||
|         args: | ||||
|           - "--tab" | ||||
|   - repo: https://github.com/shellcheck-py/shellcheck-py | ||||
|     rev: "v0.10.0.1" | ||||
|     rev: "v0.11.0.1" | ||||
|     hooks: | ||||
|       - id: shellcheck | ||||
|   - repo: https://github.com/google/yamlfmt | ||||
|     rev: v0.17.2 | ||||
|     hooks: | ||||
|       - id: yamlfmt | ||||
|         exclude: "^src-ui/pnpm-lock.yaml" | ||||
|   | ||||
| @@ -2,9 +2,11 @@ | ||||
|  | ||||
| If you feel like contributing to the project, please do! Bug fixes and improvements are always welcome. | ||||
|  | ||||
| ⚠️ Please note: Pull requests that implement a new feature or enhancement _should almost always target an existing feature request_ with evidence of community interest and discussion. This is in order to balance the work of implementing and maintaining new features / enhancements. Pull requests that are opened without meeting this requirement may not be merged. | ||||
|  | ||||
| If you want to implement something big: | ||||
|  | ||||
| - Please start a discussion about that in the issues! Maybe something similar is already in development and we can make it happen together. | ||||
| - As above, please start with a discussion! Maybe something similar is already in development and we can make it happen together. | ||||
| - When making additions to the project, consider if the majority of users will benefit from your change. If not, you're probably better of forking the project. | ||||
| - Also consider if your change will get in the way of other users. A good change is a change that enhances the experience of some users who want that change and does not affect users who do not care about the change. | ||||
| - Please see the [paperless-ngx merge process](#merging-prs) below. | ||||
| @@ -37,6 +39,8 @@ Before you can run `pytest`, ensure to [properly set up your local environment]( | ||||
|  | ||||
| Once you have submitted a **P**ull **R**equest it will be reviewed, approved, and merged by one or more community members of any team. Automated code tests and formatting checks must be passed. | ||||
|  | ||||
| Important: Pull requests that implement a new feature or enhancement _should almost always target an existing feature request_ with evidence of community interest and discussion. This is in order to balance the work of implementing and maintaining new features / enhancements. Instead of opening a PR which does not meet this requirement, please open a feature request instead, to gather feedback from both users and the project maintainers. | ||||
|  | ||||
| ## Non-Trivial Requests | ||||
|  | ||||
| PRs deemed `non-trivial` will go through a stricter review process before being merged into `dev`. This is to ensure code quality and complete functionality (free of side effects). | ||||
| @@ -81,7 +85,7 @@ Some notes about translation: | ||||
|  | ||||
| If a language has already been added, and you would like to contribute new translations or change existing translations, please read the "Translation" section in the README.md file for further details on that. | ||||
|  | ||||
| If you would like the project to be translated to another language, first head over to https://crwd.in/paperless-ngx to check if that language has already been enabled for translation. | ||||
| If you would like the project to be translated to another language, first head over to https://crowdin.com/project/paperless-ngx to check if that language has already been enabled for translation. | ||||
| If not, please request the language to be added by creating an issue on GitHub. The issue should contain: | ||||
|  | ||||
| - English name of the language (the localized name can be added on Crowdin). | ||||
| @@ -109,28 +113,12 @@ Paperless-ngx is a community project. We do our best to delegate permission and | ||||
|  | ||||
| ## Structure | ||||
|  | ||||
| As of writing, there are 21 members in paperless-ngx. 4 of these people have complete administrative privileges to the repo: | ||||
| There are currently 2 members in paperless-ngx with complete administrative privileges to the repo: | ||||
|  | ||||
| - [@shamoon](https://github.com/shamoon) | ||||
| - [@bauerj](https://github.com/bauerj) | ||||
| - [@qcasey](https://github.com/qcasey) | ||||
| - [@FrankStrieter](https://github.com/FrankStrieter) | ||||
| - [@stumpylog](https://github.com/stumpylog) | ||||
|  | ||||
| There are 5 teams collaborating on specific tasks within paperless-ngx: | ||||
|  | ||||
| - @paperless-ngx/backend (Python / django) | ||||
| - @paperless-ngx/frontend (JavaScript / Typescript) | ||||
| - @paperless-ngx/ci-cd (GitHub Actions / Deployment) | ||||
| - @paperless-ngx/issues (Issue triage) | ||||
| - @paperless-ngx/test (General testing for larger PRs) | ||||
|  | ||||
| ## Permissions | ||||
|  | ||||
| All team members are notified when mentioned or assigned to a relevant issue or pull request. Additionally, each team has slightly different access to paperless-ngx: | ||||
|  | ||||
| - The **test** team has no special permissions. | ||||
| - The **issues** team has `triage` access. This means they can organize issues and pull requests. | ||||
| - The **backend**, **frontend**, and **ci-cd** teams have `write` access. This means they can approve PRs and push code, containers, releases, and more. | ||||
| There are other members who occasionally contribute but we are actively seeking more dedicated maintainers of the project. Please reach out if you are interested. | ||||
|  | ||||
| ## Joining | ||||
|  | ||||
| @@ -141,13 +129,13 @@ The admins occasionally invite contributors directly if we believe having them o | ||||
| # 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: | ||||
| community members. That said, in an effort to keep the repository organized and manageable 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: 180 days of inactivity, < 5 "up-votes" after 180 days, < 20 "up-votes" after 1 year or < 80 "up-votes" at 2 years. | ||||
| - Feature requests that do not meet the following thresholds will be closed: 180 days of inactivity with less than 80 "up-votes", < 5 "up-votes" after 180 days, < 20 "up-votes" after 1 year or < 40 "up-votes" at 2 years. | ||||
|  | ||||
| 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. | ||||
|   | ||||
| @@ -21,7 +21,7 @@ ARG PNGX_TAG_VERSION= | ||||
| RUN set -eux && \ | ||||
| case "${PNGX_TAG_VERSION}" in \ | ||||
|   dev|beta|fix*|feature*) \ | ||||
|     sed -i -E "s/version: '([0-9\.]+)'/version: '\1 #${PNGX_TAG_VERSION}'/g" /src/src-ui/src/environments/environment.prod.ts \ | ||||
|     sed -i -E "s/tag: '([a-z\.]+)'/tag: '${PNGX_TAG_VERSION}'/g" /src/src-ui/src/environments/environment.prod.ts \ | ||||
|     ;; \ | ||||
| esac | ||||
|  | ||||
| @@ -32,7 +32,7 @@ RUN set -eux \ | ||||
| # Purpose: Installs s6-overlay and rootfs | ||||
| # Comments: | ||||
| #  - Don't leave anything extra in here either | ||||
| FROM ghcr.io/astral-sh/uv:0.6.5-python3.12-bookworm-slim AS s6-overlay-base | ||||
| FROM ghcr.io/astral-sh/uv:0.8.17-python3.12-bookworm-slim AS s6-overlay-base | ||||
|  | ||||
| WORKDIR /usr/src/s6 | ||||
|  | ||||
| @@ -47,7 +47,7 @@ ENV \ | ||||
| ARG TARGETARCH | ||||
| ARG TARGETVARIANT | ||||
| # Lock this version | ||||
| ARG S6_OVERLAY_VERSION=3.2.0.2 | ||||
| ARG S6_OVERLAY_VERSION=3.2.1.0 | ||||
|  | ||||
| ARG S6_BUILD_TIME_PKGS="curl \ | ||||
|                         xz-utils" | ||||
| @@ -239,6 +239,7 @@ COPY --from=compile-frontend --chown=1000:1000 /src/src/documents/static/fronten | ||||
| # add users, setup scripts | ||||
| # Mount the compiled frontend to expected location | ||||
| RUN set -eux \ | ||||
|   && sed -i '1s|^#!/usr/bin/env python3|#!/command/with-contenv python3|' manage.py \ | ||||
|   && echo "Setting up user/group" \ | ||||
|     && addgroup --gid 1000 paperless \ | ||||
|     && useradd --uid 1000 --gid paperless --home-dir /usr/src/paperless paperless \ | ||||
| @@ -264,4 +265,4 @@ ENTRYPOINT ["/init"] | ||||
|  | ||||
| EXPOSE 8000 | ||||
|  | ||||
| HEALTHCHECK --interval=30s --timeout=10s --retries=5 CMD [ "curl", "-fs", "-S", "--max-time", "2", "http://localhost:8000" ] | ||||
| HEALTHCHECK --interval=30s --timeout=10s --retries=5 CMD [ "curl", "-fs", "-S", "-L", "--max-time", "2", "http://localhost:8000" ] | ||||
|   | ||||
| @@ -83,7 +83,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. If you want to help out by translating paperless-ngx into your language, please head over to https://crwd.in/paperless-ngx, and thank you! More details can be found in [CONTRIBUTING.md](https://github.com/paperless-ngx/paperless-ngx/blob/main/CONTRIBUTING.md#translating-paperless-ngx). | ||||
| Paperless-ngx is available in many languages that are coordinated on Crowdin. If you want to help out by translating paperless-ngx into your language, please head over to https://crowdin.com/project/paperless-ngx, and thank you! More details can be found in [CONTRIBUTING.md](https://github.com/paperless-ngx/paperless-ngx/blob/main/CONTRIBUTING.md#translating-paperless-ngx). | ||||
|  | ||||
| ## Feature Requests | ||||
|  | ||||
|   | ||||
| @@ -1,11 +1,10 @@ | ||||
| # 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 necessary containers with the | ||||
| # correct networking for the tests | ||||
|  | ||||
| services: | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:8.17 | ||||
|     image: docker.io/gotenberg/gotenberg:8.23 | ||||
|     hostname: gotenberg | ||||
|     container_name: gotenberg | ||||
|     network_mode: host | ||||
|   | ||||
| @@ -32,6 +32,6 @@ | ||||
| # Note that this is different from PAPERLESS_OCR_LANGUAGE (default=eng), which defines | ||||
| # the language used for OCR. | ||||
| # The container installs English, German, Italian, Spanish and French by default. | ||||
| # See https://packages.debian.org/search?keywords=tesseract-ocr-&searchon=names&suite=buster | ||||
| # See https://packages.debian.org/search?keywords=tesseract-ocr-&searchon=names | ||||
| # for available languages. | ||||
| #PAPERLESS_OCR_LANGUAGES=tur ces | ||||
|   | ||||
| @@ -16,29 +16,26 @@ | ||||
| # - Instead of SQLite (default), MariaDB is used as the database server. | ||||
| # - Apache Tika and Gotenberg servers are started with paperless and paperless | ||||
| #   is configured to use these services. These provide support for consuming | ||||
| #   Office documents (Word, Excel, Power Point and their LibreOffice counter- | ||||
| #   parts. | ||||
| #   Office documents (Word, Excel, PowerPoint and their LibreOffice counter- | ||||
| #   parts). | ||||
| # | ||||
| # To install and update paperless with this file, do the following: | ||||
| # | ||||
| # - 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'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
|  | ||||
| services: | ||||
|   broker: | ||||
|     image: docker.io/library/redis:7 | ||||
|     image: docker.io/library/redis:8 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   db: | ||||
|     image: docker.io/library/mariadb:11 | ||||
|     image: docker.io/library/mariadb:12 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - dbdata:/var/lib/mysql | ||||
| @@ -48,7 +45,6 @@ services: | ||||
|       MARIADB_USER: paperless | ||||
|       MARIADB_PASSWORD: paperless | ||||
|       MARIADB_ROOT_PASSWORD: paperless | ||||
|  | ||||
|   webserver: | ||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
|     restart: unless-stopped | ||||
| @@ -75,9 +71,8 @@ services: | ||||
|       PAPERLESS_TIKA_ENABLED: 1 | ||||
|       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
|  | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:8.17 | ||||
|     image: docker.io/gotenberg/gotenberg:8.23 | ||||
|     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. | ||||
| @@ -85,11 +80,9 @@ services: | ||||
|       - "gotenberg" | ||||
|       - "--chromium-disable-javascript=true" | ||||
|       - "--chromium-allow-list=file:///tmp/.*" | ||||
|  | ||||
|   tika: | ||||
|     image: docker.io/apache/tika:latest | ||||
|     restart: unless-stopped | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -20,21 +20,18 @@ | ||||
| # - 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'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
|  | ||||
| services: | ||||
|   broker: | ||||
|     image: docker.io/library/redis:7 | ||||
|     image: docker.io/library/redis:8 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   db: | ||||
|     image: docker.io/library/mariadb:11 | ||||
|     image: docker.io/library/mariadb:12 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - dbdata:/var/lib/mysql | ||||
| @@ -44,7 +41,6 @@ services: | ||||
|       MARIADB_USER: paperless | ||||
|       MARIADB_PASSWORD: paperless | ||||
|       MARIADB_ROOT_PASSWORD: paperless | ||||
|  | ||||
|   webserver: | ||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
|     restart: unless-stopped | ||||
| @@ -66,7 +62,6 @@ services: | ||||
|       PAPERLESS_DBUSER: paperless # only needed if non-default username | ||||
|       PAPERLESS_DBPASS: paperless # only needed if non-default password | ||||
|       PAPERLESS_DBPORT: 3306 | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -22,21 +22,15 @@ | ||||
| # - Upload 'docker-compose.env' by clicking on 'Load variables from .env file' | ||||
| # - Modify the environment variables as needed | ||||
| # - Click 'Deploy the stack' and wait for it to be deployed | ||||
| # - Open the list of containers, select paperless_webserver_1 | ||||
| # - Click 'Console' and then 'Connect' to open the command line inside the container | ||||
| # - Run 'python3 manage.py createsuperuser' to create a user | ||||
| # - Exit the console | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
|  | ||||
| services: | ||||
|   broker: | ||||
|     image: docker.io/library/redis:7 | ||||
|     image: docker.io/library/redis:8 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   db: | ||||
|     image: docker.io/library/postgres:17 | ||||
|     restart: unless-stopped | ||||
| @@ -46,7 +40,6 @@ services: | ||||
|       POSTGRES_DB: paperless | ||||
|       POSTGRES_USER: paperless | ||||
|       POSTGRES_PASSWORD: paperless | ||||
|  | ||||
|   webserver: | ||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
|     restart: unless-stopped | ||||
| @@ -65,7 +58,6 @@ services: | ||||
|       PAPERLESS_DBHOST: db | ||||
|     env_file: | ||||
|       - stack.env | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -16,27 +16,24 @@ | ||||
| # - Instead of SQLite (default), PostgreSQL is used as the database server. | ||||
| # - Apache Tika and Gotenberg servers are started with paperless and paperless | ||||
| #   is configured to use these services. These provide support for consuming | ||||
| #   Office documents (Word, Excel, Power Point and their LibreOffice counter- | ||||
| #   parts. | ||||
| #   Office documents (Word, Excel, PowerPoint and their LibreOffice counter- | ||||
| #   parts). | ||||
| # | ||||
| # To install and update paperless with this file, do the following: | ||||
| # | ||||
| # - 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'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
|  | ||||
| services: | ||||
|   broker: | ||||
|     image: docker.io/library/redis:7 | ||||
|     image: docker.io/library/redis:8 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   db: | ||||
|     image: docker.io/library/postgres:17 | ||||
|     restart: unless-stopped | ||||
| @@ -46,7 +43,6 @@ services: | ||||
|       POSTGRES_DB: paperless | ||||
|       POSTGRES_USER: paperless | ||||
|       POSTGRES_PASSWORD: paperless | ||||
|  | ||||
|   webserver: | ||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
|     restart: unless-stopped | ||||
| @@ -69,22 +65,18 @@ services: | ||||
|       PAPERLESS_TIKA_ENABLED: 1 | ||||
|       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
|  | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:8.17 | ||||
|     image: docker.io/gotenberg/gotenberg:8.23 | ||||
|     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. | ||||
|     command: | ||||
|       - "gotenberg" | ||||
|       - "--chromium-disable-javascript=true" | ||||
|       - "--chromium-allow-list=file:///tmp/.*" | ||||
|  | ||||
|   tika: | ||||
|     image: docker.io/apache/tika:latest | ||||
|     restart: unless-stopped | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -20,19 +20,16 @@ | ||||
| # - 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'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
|  | ||||
| services: | ||||
|   broker: | ||||
|     image: docker.io/library/redis:7 | ||||
|     image: docker.io/library/redis:8 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   db: | ||||
|     image: docker.io/library/postgres:17 | ||||
|     restart: unless-stopped | ||||
| @@ -42,7 +39,6 @@ services: | ||||
|       POSTGRES_DB: paperless | ||||
|       POSTGRES_USER: paperless | ||||
|       POSTGRES_PASSWORD: paperless | ||||
|  | ||||
|   webserver: | ||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
|     restart: unless-stopped | ||||
| @@ -60,7 +56,6 @@ services: | ||||
|     environment: | ||||
|       PAPERLESS_REDIS: redis://broker:6379 | ||||
|       PAPERLESS_DBHOST: db | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -16,27 +16,24 @@ | ||||
| # | ||||
| # - Apache Tika and Gotenberg servers are started with paperless and paperless | ||||
| #   is configured to use these services. These provide support for consuming | ||||
| #   Office documents (Word, Excel, Power Point and their LibreOffice counter- | ||||
| #   parts. | ||||
| #   Office documents (Word, Excel, PowerPoint and their LibreOffice counter- | ||||
| #   parts). | ||||
| # | ||||
| # To install and update paperless with this file, do the following: | ||||
| # | ||||
| # - 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'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
|  | ||||
| services: | ||||
|   broker: | ||||
|     image: docker.io/library/redis:7 | ||||
|     image: docker.io/library/redis:8 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   webserver: | ||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
|     restart: unless-stopped | ||||
| @@ -57,22 +54,18 @@ services: | ||||
|       PAPERLESS_TIKA_ENABLED: 1 | ||||
|       PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 | ||||
|       PAPERLESS_TIKA_ENDPOINT: http://tika:9998 | ||||
|  | ||||
|   gotenberg: | ||||
|     image: docker.io/gotenberg/gotenberg:8.17 | ||||
|     image: docker.io/gotenberg/gotenberg:8.23 | ||||
|     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. | ||||
|     command: | ||||
|       - "gotenberg" | ||||
|       - "--chromium-disable-javascript=true" | ||||
|       - "--chromium-allow-list=file:///tmp/.*" | ||||
|  | ||||
|   tika: | ||||
|     image: docker.io/apache/tika:latest | ||||
|     restart: unless-stopped | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -17,19 +17,16 @@ | ||||
| # - 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'. | ||||
| # | ||||
| # For more extensive installation and update instructions, refer to the | ||||
| # documentation. | ||||
|  | ||||
| services: | ||||
|   broker: | ||||
|     image: docker.io/library/redis:7 | ||||
|     image: docker.io/library/redis:8 | ||||
|     restart: unless-stopped | ||||
|     volumes: | ||||
|       - redisdata:/data | ||||
|  | ||||
|   webserver: | ||||
|     image: ghcr.io/paperless-ngx/paperless-ngx:latest | ||||
|     restart: unless-stopped | ||||
| @@ -45,7 +42,6 @@ services: | ||||
|     env_file: docker-compose.env | ||||
|     environment: | ||||
|       PAPERLESS_REDIS: redis://broker:6379 | ||||
|  | ||||
| volumes: | ||||
|   data: | ||||
|   media: | ||||
|   | ||||
| @@ -1,179 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| set -e | ||||
|  | ||||
| # Source: https://github.com/sameersbn/docker-gitlab/ | ||||
| map_uidgid() { | ||||
| 	local -r usermap_original_uid=$(id -u paperless) | ||||
| 	local -r usermap_original_gid=$(id -g paperless) | ||||
| 	local -r usermap_new_uid=${USERMAP_UID:-$usermap_original_uid} | ||||
| 	local -r usermap_new_gid=${USERMAP_GID:-${usermap_original_gid:-$usermap_new_uid}} | ||||
| 	if [[ ${usermap_new_uid} != "${usermap_original_uid}" || ${usermap_new_gid} != "${usermap_original_gid}" ]]; then | ||||
| 		echo "Mapping UID and GID for paperless:paperless to $usermap_new_uid:$usermap_new_gid" | ||||
| 		usermod --non-unique --uid "${usermap_new_uid}" paperless | ||||
| 		groupmod --non-unique --gid "${usermap_new_gid}" paperless | ||||
| 	fi | ||||
| } | ||||
|  | ||||
| map_folders() { | ||||
| 	# Export these so they can be used in docker-prepare.sh | ||||
| 	export DATA_DIR="${PAPERLESS_DATA_DIR:-/usr/src/paperless/data}" | ||||
| 	export MEDIA_ROOT_DIR="${PAPERLESS_MEDIA_ROOT:-/usr/src/paperless/media}" | ||||
| 	export CONSUME_DIR="${PAPERLESS_CONSUMPTION_DIR:-/usr/src/paperless/consume}" | ||||
| } | ||||
|  | ||||
| custom_container_init() { | ||||
| 	# Mostly borrowed from the LinuxServer.io base image | ||||
| 	# https://github.com/linuxserver/docker-baseimage-ubuntu/tree/bionic/root/etc/cont-init.d | ||||
| 	local -r custom_script_dir="/custom-cont-init.d" | ||||
| 	# Tamper checking. | ||||
| 	# Don't run files which are owned by anyone except root | ||||
| 	# Don't run files which are writeable by others | ||||
| 	if [ -d "${custom_script_dir}" ]; then | ||||
| 		if [ -n "$(/usr/bin/find "${custom_script_dir}" -maxdepth 1 ! -user root)" ]; then | ||||
| 			echo "**** Potential tampering with custom scripts detected ****" | ||||
| 			echo "**** The folder '${custom_script_dir}' must be owned by root ****" | ||||
| 			return 0 | ||||
| 		fi | ||||
| 		if [ -n "$(/usr/bin/find "${custom_script_dir}" -maxdepth 1 -perm -o+w)" ]; then | ||||
| 			echo "**** The folder '${custom_script_dir}' or some of contents have write permissions for others, which is a security risk. ****" | ||||
| 			echo "**** Please review the permissions and their contents to make sure they are owned by root, and can only be modified by root. ****" | ||||
| 			return 0 | ||||
| 		fi | ||||
|  | ||||
| 		# Make sure custom init directory has files in it | ||||
| 		if [ -n "$(/bin/ls --almost-all "${custom_script_dir}" 2>/dev/null)" ]; then | ||||
| 			echo "[custom-init] files found in ${custom_script_dir} executing" | ||||
| 			# Loop over files in the directory | ||||
| 			for SCRIPT in "${custom_script_dir}"/*; do | ||||
| 				NAME="$(basename "${SCRIPT}")" | ||||
| 				if [ -f "${SCRIPT}" ]; then | ||||
| 					echo "[custom-init] ${NAME}: executing..." | ||||
| 					/bin/bash "${SCRIPT}" | ||||
| 					echo "[custom-init] ${NAME}: exited $?" | ||||
| 				elif [ ! -f "${SCRIPT}" ]; then | ||||
| 					echo "[custom-init] ${NAME}: is not a file" | ||||
| 				fi | ||||
| 			done | ||||
| 		else | ||||
| 			echo "[custom-init] no custom files found exiting..." | ||||
| 		fi | ||||
|  | ||||
| 	fi | ||||
| } | ||||
|  | ||||
| initialize() { | ||||
|  | ||||
| 	# Setup environment from secrets before anything else | ||||
| 	# Check for a version of this var with _FILE appended | ||||
| 	# and convert the contents to the env var value | ||||
| 	# Source it so export is persistent | ||||
| 	# shellcheck disable=SC1091 | ||||
| 	source /sbin/env-from-file.sh | ||||
|  | ||||
| 	# Change the user and group IDs if needed | ||||
| 	map_uidgid | ||||
|  | ||||
| 	# Check for overrides of certain folders | ||||
| 	map_folders | ||||
|  | ||||
| 	local -r export_dir="/usr/src/paperless/export" | ||||
|  | ||||
| 	for dir in \ | ||||
| 		"${export_dir}" \ | ||||
| 		"${DATA_DIR}" "${DATA_DIR}/index" \ | ||||
| 		"${MEDIA_ROOT_DIR}" "${MEDIA_ROOT_DIR}/documents" "${MEDIA_ROOT_DIR}/documents/originals" "${MEDIA_ROOT_DIR}/documents/thumbnails" \ | ||||
| 		"${CONSUME_DIR}"; do | ||||
| 		if [[ ! -d "${dir}" ]]; then | ||||
| 			echo "Creating directory ${dir}" | ||||
| 			mkdir --parents --verbose "${dir}" | ||||
| 		fi | ||||
| 	done | ||||
|  | ||||
| 	local -r tmp_dir="${PAPERLESS_SCRATCH_DIR:=/tmp/paperless}" | ||||
| 	echo "Creating directory scratch directory ${tmp_dir}" | ||||
| 	mkdir --parents --verbose "${tmp_dir}" | ||||
|  | ||||
| 	set +e | ||||
| 	echo "Adjusting permissions of paperless files. This may take a while." | ||||
| 	chown -R paperless:paperless "${tmp_dir}" | ||||
| 	for dir in \ | ||||
| 		"${export_dir}" \ | ||||
| 		"${DATA_DIR}" \ | ||||
| 		"${MEDIA_ROOT_DIR}" \ | ||||
| 		"${CONSUME_DIR}"; do | ||||
| 		find "${dir}" -not \( -user paperless -and -group paperless \) -exec chown --changes paperless:paperless {} + | ||||
| 	done | ||||
| 	set -e | ||||
|  | ||||
| 	"${gosu_cmd[@]}" /sbin/docker-prepare.sh | ||||
|  | ||||
| 	# Leave this last thing | ||||
| 	custom_container_init | ||||
|  | ||||
| } | ||||
|  | ||||
| install_languages() { | ||||
| 	echo "Installing languages..." | ||||
|  | ||||
| 	read -ra langs <<<"$1" | ||||
|  | ||||
| 	# Check that it is not empty | ||||
| 	if [ ${#langs[@]} -eq 0 ]; then | ||||
| 		return | ||||
| 	fi | ||||
|  | ||||
| 	# Build list of packages to install | ||||
| 	to_install=() | ||||
| 	for lang in "${langs[@]}"; do | ||||
| 		pkg="tesseract-ocr-$lang" | ||||
|  | ||||
| 		if dpkg --status "$pkg" &>/dev/null; then | ||||
| 			echo "Package $pkg already installed!" | ||||
| 			continue | ||||
| 		else | ||||
| 			to_install+=("$pkg") | ||||
| 		fi | ||||
| 	done | ||||
|  | ||||
| 	# Use apt only when we install packages | ||||
| 	if [ ${#to_install[@]} -gt 0 ]; then | ||||
| 		apt-get update | ||||
|  | ||||
| 		for pkg in "${to_install[@]}"; do | ||||
|  | ||||
| 			if ! apt-cache show "$pkg" &>/dev/null; then | ||||
| 				echo "Skipped $pkg: Package not found! :(" | ||||
| 				continue | ||||
| 			fi | ||||
|  | ||||
| 			echo "Installing package $pkg..." | ||||
| 			if ! apt-get --assume-yes install "$pkg" &>/dev/null; then | ||||
| 				echo "Could not install $pkg" | ||||
| 				exit 1 | ||||
| 			fi | ||||
| 		done | ||||
| 	fi | ||||
| } | ||||
|  | ||||
| echo "Paperless-ngx docker container starting..." | ||||
|  | ||||
| gosu_cmd=(gosu paperless) | ||||
| if [ "$(id --user)" == "$(id --user paperless)" ]; then | ||||
| 	gosu_cmd=() | ||||
| fi | ||||
|  | ||||
| # Install additional languages if specified | ||||
| if [[ -n "$PAPERLESS_OCR_LANGUAGES" ]]; then | ||||
| 	install_languages "$PAPERLESS_OCR_LANGUAGES" | ||||
| fi | ||||
|  | ||||
| initialize | ||||
|  | ||||
| if [[ "$1" != "/"* ]]; then | ||||
| 	echo Executing management command "$@" | ||||
| 	exec "${gosu_cmd[@]}" python3 manage.py "$@" | ||||
| else | ||||
| 	echo Executing "$@" | ||||
| 	exec "$@" | ||||
| fi | ||||
| @@ -18,9 +18,10 @@ for command in decrypt_documents \ | ||||
| 	document_fuzzy_match \ | ||||
| 	manage_superuser \ | ||||
| 	convert_mariadb_uuid \ | ||||
| 	prune_audit_logs; | ||||
| 	prune_audit_logs \ | ||||
| 	createsuperuser; | ||||
| do | ||||
| 	echo "installing $command..." | ||||
| 	sed "s/management_command/$command/g" management_script.sh >"$PWD/rootfs/usr/local/bin/$command" | ||||
| 	chmod +x "$PWD/rootfs/usr/local/bin/$command" | ||||
| 	chmod u=rwx,g=rwx,o=rx "$PWD/rootfs/usr/local/bin/$command" | ||||
| done | ||||
|   | ||||
| @@ -9,7 +9,7 @@ if find /run/s6/container_environment/*"_FILE" -maxdepth 1 > /dev/null 2>&1; the | ||||
| 	for FILENAME in /run/s6/container_environment/*; do | ||||
| 		if [[ "${FILENAME##*/}" == PAPERLESS_*_FILE ]]; then | ||||
| 			# This should have been named different.. | ||||
| 			if [[ ${FILENAME} == "PAPERLESS_OCR_SKIP_ARCHIVE_FILE" || ${FILENAME} == "PAPERLESS_MODEL_FILE" ]]; then | ||||
| 			if [[ "${FILENAME##*/}" == "PAPERLESS_OCR_SKIP_ARCHIVE_FILE" || "${FILENAME##*/}" == "PAPERLESS_MODEL_FILE" ]]; then | ||||
| 				continue | ||||
| 			fi | ||||
| 			SECRETFILE=$(cat "${FILENAME}") | ||||
| @@ -17,6 +17,9 @@ if find /run/s6/container_environment/*"_FILE" -maxdepth 1 > /dev/null 2>&1; the | ||||
| 			if [[ -f ${SECRETFILE} ]]; then | ||||
| 				# Trim off trailing _FILE | ||||
| 				FILESTRIP=${FILENAME//_FILE/} | ||||
| 				if [[ $(tail -n1 "${SECRETFILE}" | wc -l) != 0 ]]; then | ||||
| 					echo "${log_prefix} Your secret: ${FILENAME##*/} contains a trailing newline and may not work as expected" | ||||
| 				fi | ||||
| 				# Set environment variable | ||||
| 				cat "${SECRETFILE}" > "${FILESTRIP}" | ||||
| 				echo "${log_prefix} ${FILESTRIP##*/} set from ${FILENAME##*/}" | ||||
|   | ||||
| @@ -9,25 +9,57 @@ declare -r media_root_dir="${PAPERLESS_MEDIA_ROOT:-/usr/src/paperless/media}" | ||||
| declare -r consume_dir="${PAPERLESS_CONSUMPTION_DIR:-/usr/src/paperless/consume}" | ||||
| declare -r tmp_dir="${PAPERLESS_SCRATCH_DIR:=/tmp/paperless}" | ||||
|  | ||||
| echo "${log_prefix} Checking for folder existence" | ||||
| declare -r main_dirs=( | ||||
| 	"${export_dir}" | ||||
| 	"${data_dir}" | ||||
| 	"${media_root_dir}" | ||||
| 	"${consume_dir}" | ||||
| 	"${tmp_dir}" | ||||
| ) | ||||
|  | ||||
| for dir in \ | ||||
| 	"${export_dir}" \ | ||||
| 	"${data_dir}" "${data_dir}/index" \ | ||||
| 	"${media_root_dir}" "${media_root_dir}/documents" "${media_root_dir}/documents/originals" "${media_root_dir}/documents/thumbnails" \ | ||||
| 	"${consume_dir}" \ | ||||
| 	"${tmp_dir}"; do | ||||
| 	if [[ ! -d "${dir}" ]]; then | ||||
| 		mkdir --parents --verbose "${dir}" | ||||
| 	fi | ||||
| done | ||||
| declare -r extra_dirs=( | ||||
| 	"${main_dirs[@]}" | ||||
| 	"${data_dir}/index" | ||||
| 	"${media_root_dir}/documents" | ||||
| 	"${media_root_dir}/documents/originals" | ||||
| 	"${media_root_dir}/documents/thumbnails" | ||||
| ) | ||||
|  | ||||
| echo "${log_prefix} Adjusting file and folder permissions" | ||||
| for dir in \ | ||||
| 	"${export_dir}" \ | ||||
| 	"${data_dir}" \ | ||||
| 	"${media_root_dir}" \ | ||||
| 	"${consume_dir}" \ | ||||
| 	"${tmp_dir}"; do | ||||
| 	find "${dir}" -not \( -user paperless -and -group paperless \) -exec chown --changes paperless:paperless {} + | ||||
| done | ||||
| if [[ -n "${USER_IS_NON_ROOT}" ]]; then | ||||
| 	# Non-root mode: Create directories as current user, warn about permission issues | ||||
| 	echo "${log_prefix} Running in non-root mode, checking directories" | ||||
| 	current_uid=$(id --user) | ||||
| 	current_gid=$(id --group) | ||||
|  | ||||
| 	for dir in "${extra_dirs[@]}"; do | ||||
| 		if [[ ! -d "${dir}" ]]; then | ||||
| 			mkdir --parents --verbose "${dir}" || echo "${log_prefix} WARNING: Could not create ${dir} - permission denied" | ||||
| 		fi | ||||
| 		# Check permissions on existing directories too | ||||
| 		if [[ -d "${dir}" && ! -w "${dir}" ]]; then | ||||
| 			echo "${log_prefix} WARNING: No write permission to ${dir}" | ||||
| 		fi | ||||
| 	done | ||||
|  | ||||
| 	# Warn about ownership issues | ||||
| 	for dir in "${main_dirs[@]}"; do | ||||
| 		if [[ -d "${dir}" ]]; then | ||||
| 			find "${dir}" -not \( -user ${current_uid} -and -group ${current_gid} \) -exec echo "${log_prefix} WARNING: Permission issue on {}: not owned by current user (${current_uid}:${current_gid})" \; 2>/dev/null || echo "${log_prefix} WARNING: Cannot check permissions on ${dir}" | ||||
| 		fi | ||||
| 	done | ||||
| else | ||||
| 	# Root mode: Create and fix permissions as needed | ||||
| 	echo "${log_prefix} Running with root privileges, adjusting directories and permissions" | ||||
|  | ||||
| 	# First create directories | ||||
| 	for dir in "${extra_dirs[@]}"; do | ||||
| 		if [[ ! -d "${dir}" ]]; then | ||||
| 			mkdir --parents --verbose "${dir}" | ||||
| 		fi | ||||
| 	done | ||||
|  | ||||
| 	# Then fix permissions on all directories | ||||
| 	for dir in "${main_dirs[@]}"; do | ||||
| 		find "${dir}" -not \( -user paperless -and -group paperless \) -exec chown --changes paperless:paperless {} + | ||||
| 	done | ||||
| fi | ||||
|   | ||||
| @@ -1,20 +1,18 @@ | ||||
| #!/command/with-contenv /usr/bin/bash | ||||
| # shellcheck shell=bash | ||||
| declare -r log_prefix="[init-migrations]" | ||||
|  | ||||
| declare -r data_dir="${PAPERLESS_DATA_DIR:-/usr/src/paperless/data}" | ||||
|  | ||||
| ( | ||||
| 	# flock is in place to prevent multiple containers from doing migrations | ||||
| 	# simultaneously. This also ensures that the db is ready when the command | ||||
| 	# of the current container starts. | ||||
| 	flock 200 | ||||
| 	echo "${log_prefix} Apply database migrations..." | ||||
| 	cd "${PAPERLESS_SRC_DIR}" | ||||
| echo "${log_prefix} Apply database migrations..." | ||||
|  | ||||
| 	if [[ -n "${USER_IS_NON_ROOT}" ]]; then | ||||
| 		exec python3 manage.py migrate --skip-checks --no-input | ||||
| 	else | ||||
| 		exec s6-setuidgid paperless python3 manage.py migrate --skip-checks --no-input | ||||
| 	fi | ||||
| cd "${PAPERLESS_SRC_DIR}" | ||||
|  | ||||
| ) 200>"${data_dir}/migration_lock" | ||||
| # The whole migrate, with flock, needs to run as the right user | ||||
| if [[ -n "${USER_IS_NON_ROOT}" ]]; then | ||||
| 	exec s6-setlock -n "${data_dir}/migration_lock" python3 manage.py migrate --skip-checks --no-input | ||||
| else | ||||
| 	exec s6-setuidgid paperless \ | ||||
| 		s6-setlock -n "${data_dir}/migration_lock" \ | ||||
| 		python3 manage.py migrate --skip-checks --no-input | ||||
| fi | ||||
|   | ||||
| @@ -11,9 +11,10 @@ printf "/usr/src/paperless/src" > /var/run/s6/container_environment/PAPERLESS_SR | ||||
| echo $(date +%s) > /var/run/s6/container_environment/PAPERLESS_START_TIME_S | ||||
|  | ||||
| # Check if we're starting as a non-root user | ||||
| if [ $(id -u) == $(id -u paperless) ]; then | ||||
| if [ "$(id --user)" != "0" ]; then | ||||
| 	printf "true" > /var/run/s6/container_environment/USER_IS_NON_ROOT | ||||
| 	echo "${log_prefix}  paperless-ngx docker container running under a user" | ||||
| 	echo "${log_prefix} paperless-ngx docker container running under a user ($(id --user):$(id --group))" | ||||
| else | ||||
| 	echo "${log_prefix}  paperless-ngx docker container starting init as root" | ||||
| 	printf "/usr/src/paperless" > /var/run/s6/container_environment/HOME | ||||
| 	echo "${log_prefix} paperless-ngx docker container starting init as root" | ||||
| fi | ||||
|   | ||||
| @@ -14,7 +14,7 @@ if [[ -n "${PAPERLESS_FORCE_SCRIPT_NAME}" ]]; then | ||||
| fi | ||||
|  | ||||
| if [[ -n "${USER_IS_NON_ROOT}" ]]; then | ||||
|   exec granian --interface asginl --ws "paperless.asgi:application" | ||||
|   exec granian --interface asginl --ws --loop uvloop "paperless.asgi:application" | ||||
| else | ||||
|   exec s6-setuidgid paperless granian --interface asginl --ws "paperless.asgi:application" | ||||
|   exec s6-setuidgid paperless granian --interface asginl --ws --loop uvloop "paperless.asgi:application" | ||||
| fi | ||||
|   | ||||
							
								
								
									
										14
									
								
								docker/rootfs/usr/local/bin/createsuperuser
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										14
									
								
								docker/rootfs/usr/local/bin/createsuperuser
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| #!/command/with-contenv /usr/bin/bash | ||||
| # shellcheck shell=bash | ||||
|  | ||||
| set -e | ||||
|  | ||||
| cd "${PAPERLESS_SRC_DIR}" | ||||
|  | ||||
| if [[ $(id -u) == 0 ]]; then | ||||
| 	s6-setuidgid paperless python3 manage.py createsuperuser "$@" | ||||
| elif [[ $(id -un) == "paperless" ]]; then | ||||
| 	python3 manage.py createsuperuser "$@" | ||||
| else | ||||
| 	echo "Unknown user." | ||||
| fi | ||||
| @@ -179,10 +179,14 @@ following: | ||||
|  | ||||
| ### Database Upgrades | ||||
|  | ||||
| In general, paperless does not require a specific version of PostgreSQL or MariaDB and it is | ||||
| Paperless-ngx is compatible with Django-supported versions of PostgreSQL and MariaDB and it is generally | ||||
| 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. | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     As of Paperless-ngx v2.18, the minimum supported version of PostgreSQL is 14. | ||||
|  | ||||
| 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/) | ||||
| @@ -306,7 +310,7 @@ in dedicated folders according to their nature: `archive`, `originals`, | ||||
| If `-sm` or `--split-manifest` is provided, information about document | ||||
| will be placed in individual json files, instead of a single JSON file. The main | ||||
| manifest.json will still contain application wide information (e.g. tags, correspondent, | ||||
| documenttype, etc) | ||||
| document type, etc) | ||||
|  | ||||
| If `-z` or `--zip` is provided, the export will be a zip file | ||||
| in the target directory, named according to the current local date or the | ||||
| @@ -333,7 +337,7 @@ must be provided to import. If this value is lost, the export cannot be imported | ||||
| The document importer takes the export produced by the [Document | ||||
| exporter](#exporter) and imports it into paperless. | ||||
|  | ||||
| The importer works just like the exporter. You point it at a directory, | ||||
| The importer works just like the exporter. You point it at a directory or the generated .zip file, | ||||
| and the script does the rest of the work: | ||||
|  | ||||
| ```shell | ||||
| @@ -351,9 +355,6 @@ 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. You must unzip them into | ||||
| the target directory first. | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     Importing from a previous version of Paperless may work, but for best | ||||
| @@ -460,6 +461,22 @@ of the index and usually makes queries faster and also ensures that the | ||||
| autocompletion works properly. This command is regularly invoked by the | ||||
| task scheduler. | ||||
|  | ||||
| ### Clearing the database read cache | ||||
|  | ||||
| If the database read cache is enabled, **you must run this command** after making any changes to the database outside the application context. | ||||
| This includes operations such as restoring a database backup or executing SQL statements like UPDATE, INSERT, DELETE, ALTER, CREATE, or DROP. | ||||
|  | ||||
| Failing to invalidate the cache after such modifications can lead to stale data being served from the cache, and **may cause data corruption** or inconsistent behavior in the application. | ||||
|  | ||||
| Use the following management command to clear the cache: | ||||
|  | ||||
| ``` | ||||
| python3 manage.py invalidate_cachalot | ||||
| ``` | ||||
|  | ||||
| !!! info | ||||
| The database read cache is based on Django-Cachalot. You can refer to their [documentation](https://django-cachalot.readthedocs.io/en/latest/quickstart.html#manage-py-command). | ||||
|  | ||||
| ### Managing filenames {#renamer} | ||||
|  | ||||
| If you use paperless' feature to | ||||
| @@ -565,19 +582,15 @@ document. | ||||
|  | ||||
| ### Managing encryption {#encryption} | ||||
|  | ||||
| Documents can be stored in Paperless using GnuPG encryption. | ||||
|  | ||||
| !!! warning | ||||
|  | ||||
|     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 | ||||
|     documents is stored plain in the database, even if your documents are | ||||
|     encrypted. Filenames are not encrypted as well. | ||||
|  | ||||
|     Also, the web server provides transparent access to your encrypted | ||||
|     documents. | ||||
|     Encryption was removed in [paperless-ng 0.9](changelog.md#paperless-ng-090) | ||||
|     because it did not really provide any additional security, the passphrase | ||||
|     was stored in a configuration file on the same system as the documents. | ||||
|     Furthermore, the entire text content of the documents is stored plain in | ||||
|     the database, even if your documents are encrypted. Filenames are not | ||||
|     encrypted as well. Finally, the web server provides transparent access to | ||||
|     your encrypted documents. | ||||
|  | ||||
|     Consider running paperless on an encrypted filesystem instead, which | ||||
|     will then at least provide security against physical hardware theft. | ||||
| @@ -633,3 +646,11 @@ entries created prior to this are not removed. This command allows you to prune | ||||
| ```shell | ||||
| prune_audit_logs | ||||
| ``` | ||||
|  | ||||
| ### Create superuser {#create-superuser} | ||||
|  | ||||
| If you need to create a superuser, use the following command: | ||||
|  | ||||
| ```shell | ||||
| createsuperuser | ||||
| ``` | ||||
|   | ||||
| @@ -179,6 +179,7 @@ variables: | ||||
| | ---------------------------- | ---------------------------------------------- | | ||||
| | `DOCUMENT_ID`                | Database primary key of the document           | | ||||
| | `DOCUMENT_FILE_NAME`         | Formatted filename, not including paths        | | ||||
| | `DOCUMENT_TYPE`              | The document type (if any)                     | | ||||
| | `DOCUMENT_CREATED`           | Date & time when document created              | | ||||
| | `DOCUMENT_MODIFIED`          | Date & time when document was last modified    | | ||||
| | `DOCUMENT_ADDED`             | Date & time when document was added            | | ||||
| @@ -433,6 +434,134 @@ provided. The template is provided as a string, potentially multiline, and rende | ||||
| In addition, the entire Document instance is available to be utilized in a more advanced way, as well as some variables which only make sense to be accessed | ||||
| with more complex logic. | ||||
|  | ||||
| #### Custom Jinja2 Filters | ||||
|  | ||||
| ##### Custom Field Access | ||||
|  | ||||
| The `get_cf_value` filter retrieves a value from custom field data with optional default fallback. | ||||
|  | ||||
| ###### Syntax | ||||
|  | ||||
| ```jinja2 | ||||
| {{ custom_fields | get_cf_value('field_name') }} | ||||
| {{ custom_fields | get_cf_value('field_name', 'default_value') }} | ||||
| ``` | ||||
|  | ||||
| ###### Parameters | ||||
|  | ||||
| -   `custom_fields`: This _must_ be the provided custom field data | ||||
| -   `name` (str): Name of the custom field to retrieve | ||||
| -   `default` (str, optional): Default value to return if field is not found or has no value | ||||
|  | ||||
| ###### Returns | ||||
|  | ||||
| -   `str | None`: The field value, default value, or `None` if neither exists | ||||
|  | ||||
| ###### Examples | ||||
|  | ||||
| ```jinja2 | ||||
| <!-- Basic usage --> | ||||
| {{ custom_fields | get_cf_value('department') }} | ||||
|  | ||||
| <!-- With default value --> | ||||
| {{ custom_fields | get_cf_value('phone', 'Not provided') }} | ||||
| ``` | ||||
|  | ||||
| ##### Datetime Formatting | ||||
|  | ||||
| The `datetime` filter formats a datetime string or datetime object using Python's strftime formatting. | ||||
|  | ||||
| ###### Syntax | ||||
|  | ||||
| ```jinja2 | ||||
| {{ datetime_value | datetime('%Y-%m-%d %H:%M:%S') }} | ||||
| ``` | ||||
|  | ||||
| ###### Parameters | ||||
|  | ||||
| -   `value` (str | datetime): Date/time value to format (strings will be parsed automatically) | ||||
| -   `format` (str): Python strftime format string | ||||
|  | ||||
| ###### Returns | ||||
|  | ||||
| -   `str`: Formatted datetime string | ||||
|  | ||||
| ###### Examples | ||||
|  | ||||
| ```jinja2 | ||||
| <!-- Format datetime object --> | ||||
| {{ created | datetime('%B %d, %Y at %I:%M %p') }} | ||||
| <!-- Output: "January 15, 2024 at 02:30 PM" --> | ||||
|  | ||||
| <!-- Custom formatting --> | ||||
| {{ custom_fields | get_cf_value('Date Field') | datetime('%A, %B %d, %Y') }} | ||||
| <!-- Output: "Monday, January 15, 2024" --> | ||||
| ``` | ||||
|  | ||||
| See the [strftime format code documentation](https://docs.python.org/3.13/library/datetime.html#strftime-and-strptime-format-codes) | ||||
| for the possible codes and their meanings. | ||||
|  | ||||
| ##### Date Localization | ||||
|  | ||||
| The `localize_date` filter formats a date or datetime object into a localized string using Babel internationalization. | ||||
| This takes into account the provided locale for translation. Since this must be used on a date or datetime object, | ||||
| you must access the field directly, i.e. `document.created`. | ||||
| An ISO string can also be provided to control the output format. | ||||
|  | ||||
| ###### Syntax | ||||
|  | ||||
| ```jinja2 | ||||
| {{ date_value | localize_date('medium', 'en_US') }} | ||||
| {{ datetime_value | localize_date('short', 'fr_FR') }} | ||||
| ``` | ||||
|  | ||||
| ###### Parameters | ||||
|  | ||||
| -   `value` (date | datetime | str): Date, datetime object or ISO string to format (datetime should be timezone-aware) | ||||
| -   `format` (str): Format type - either a Babel preset ('short', 'medium', 'long', 'full') or custom pattern | ||||
| -   `locale` (str): Locale code for localization (e.g., 'en_US', 'fr_FR', 'de_DE') | ||||
|  | ||||
| ###### Returns | ||||
|  | ||||
| -   `str`: Localized, formatted date string | ||||
|  | ||||
| ###### Examples | ||||
|  | ||||
| ```jinja2 | ||||
| <!-- Preset formats --> | ||||
| {{ document.created | localize_date('short', 'en_US') }} | ||||
| <!-- Output: "1/15/24" --> | ||||
|  | ||||
| {{ document.created | localize_date('medium', 'en_US') }} | ||||
| <!-- Output: "Jan 15, 2024" --> | ||||
|  | ||||
| {{ document.created | localize_date('long', 'en_US') }} | ||||
| <!-- Output: "January 15, 2024" --> | ||||
|  | ||||
| {{ document.created | localize_date('full', 'en_US') }} | ||||
| <!-- Output: "Monday, January 15, 2024" --> | ||||
|  | ||||
| <!-- Different locales --> | ||||
| {{ document.created | localize_date('medium', 'fr_FR') }} | ||||
| <!-- Output: "15 janv. 2024" --> | ||||
|  | ||||
| {{ document.created | localize_date('medium', 'de_DE') }} | ||||
| <!-- Output: "15.01.2024" --> | ||||
|  | ||||
| <!-- Custom patterns --> | ||||
| {{ document.created | localize_date('dd/MM/yyyy', 'en_GB') }} | ||||
| <!-- Output: "15/01/2024" --> | ||||
| ``` | ||||
|  | ||||
| See the [supported format codes](https://unicode.org/reports/tr35/tr35-dates.html#Date_Format_Patterns) for more options. | ||||
|  | ||||
| ### Format Presets | ||||
|  | ||||
| -   **short**: Abbreviated format (e.g., "1/15/24") | ||||
| -   **medium**: Medium-length format (e.g., "Jan 15, 2024") | ||||
| -   **long**: Long format with full month name (e.g., "January 15, 2024") | ||||
| -   **full**: Full format including day of week (e.g., "Monday, January 15, 2024") | ||||
|  | ||||
| #### Additional Variables | ||||
|  | ||||
| -   `{{ tag_name_list }}`: A list of tag names applied to the document, ordered by the tag name. Note this is a list, not a single string | ||||
|   | ||||
							
								
								
									
										31
									
								
								docs/api.md
									
									
									
									
									
								
							
							
						
						
									
										31
									
								
								docs/api.md
									
									
									
									
									
								
							| @@ -132,7 +132,7 @@ use cases: | ||||
|  | ||||
| 5. Documents with a custom field "address" (text) that is empty: | ||||
|  | ||||
|     `?custom_field_query=["OR", ["address", "isnull", true], ["address", "exact", ""]]` | ||||
|     `?custom_field_query=["OR", [["address", "isnull", true], ["address", "exact", ""]]]` | ||||
|  | ||||
| 6. Documents that don't have a field called "foo": | ||||
|  | ||||
| @@ -192,8 +192,8 @@ The endpoint supports the following optional form fields: | ||||
| -   `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. | ||||
| -   `custom_fields`: Either an array of custom field ids to assign (with an empty | ||||
|     value) to the document or an object mapping field id -> value. | ||||
|  | ||||
| The endpoint will immediately return HTTP 200 if the document consumption | ||||
| process was started successfully, with the UUID of the consumption task | ||||
| @@ -270,7 +270,7 @@ The following methods are supported: | ||||
| -   `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] }` | ||||
|     -   Requires `parameters`: `{ "add_tags": [LIST_OF_TAG_IDS] }` and `{ "remove_tags": [LIST_OF_TAG_IDS] }` | ||||
| -   `delete` | ||||
|     -   No `parameters` required | ||||
| -   `reprocess` | ||||
| @@ -282,6 +282,18 @@ The following methods are supported: | ||||
|         -   `"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. | ||||
| -   `edit_pdf` | ||||
|     -   Requires `parameters`: | ||||
|         -   `"doc_ids": [DOCUMENT_ID]` A list of a single document ID to edit. | ||||
|         -   `"operations": [OPERATION, ...]` A list of operations to perform on the documents. Each operation is a dictionary | ||||
|             with the following keys: | ||||
|             -   `"page": PAGE_NUMBER` The page number to edit (1-based). | ||||
|             -   `"rotate": DEGREES` Optional rotation in degrees (90, 180, 270). | ||||
|             -   `"doc": OUTPUT_DOCUMENT_INDEX` Optional index of the output document for split operations. | ||||
|     -   Optional `parameters`: | ||||
|         -   `"delete_original": true` to delete the original documents after editing. | ||||
|         -   `"update_document": true` to update the existing document with the edited PDF. | ||||
|         -   `"include_metadata": true` to copy metadata from the original document to the edited document. | ||||
| -   `merge` | ||||
|     -   No additional `parameters` required. | ||||
|     -   The ordering of the merged document is determined by the list of IDs. | ||||
| @@ -413,3 +425,14 @@ Initial API version. | ||||
|     list of strings. When creating or updating a custom field value of a | ||||
|     document for a select type custom field, the value should be the `id` of | ||||
|     the option whereas previously was the index of the option. | ||||
|  | ||||
| #### Version 8 | ||||
|  | ||||
| -   The user field of document notes now returns a simplified user object | ||||
|     rather than just the user ID. | ||||
|  | ||||
| #### Version 9 | ||||
|  | ||||
| -   The document `created` field is now a date, not a datetime. The | ||||
|     `created_date` field is considered deprecated and will be removed in a | ||||
|     future version. | ||||
|   | ||||
| @@ -1,5 +1,839 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## paperless-ngx 2.18.4 | ||||
|  | ||||
| ### Features / Enhancements | ||||
|  | ||||
| -   Enhancement: report websocket status [@shamoon](https://github.com/shamoon) ([#10777](https://github.com/paperless-ngx/paperless-ngx/pull/10777)) | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Revert "Performance: Enable virtual scrolling for large custom field … [@shamoon](https://github.com/shamoon) ([#10803](https://github.com/paperless-ngx/paperless-ngx/pull/10803)) | ||||
| -   Fixhancement: update sidebar view counts on save \& next also [@shamoon](https://github.com/shamoon) ([#10793](https://github.com/paperless-ngx/paperless-ngx/pull/10793)) | ||||
| -   Performance fix: add paging for custom field select options [@shamoon](https://github.com/shamoon) ([#10755](https://github.com/paperless-ngx/paperless-ngx/pull/10755)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| <details> | ||||
| <summary>8 changes</summary> | ||||
|  | ||||
| -   Chore(deps-dev): Bump the frontend-jest-dependencies group in /src-ui with 2 updates [@shamoon](https://github.com/shamoon) ([#10770](https://github.com/paperless-ngx/paperless-ngx/pull/10770)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10745](https://github.com/paperless-ngx/paperless-ngx/pull/10745)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 22 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10744](https://github.com/paperless-ngx/paperless-ngx/pull/10744)) | ||||
| -   Chore(deps): Bump bootstrap from 5.3.7 to 5.3.8 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10740](https://github.com/paperless-ngx/paperless-ngx/pull/10740)) | ||||
| -   Chore(deps-dev): Bump @<!---->playwright/test from 1.54.2 to 1.55.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10743](https://github.com/paperless-ngx/paperless-ngx/pull/10743)) | ||||
| -   Chore(deps-dev): Bump webpack from 5.101.0 to 5.101.3 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10751](https://github.com/paperless-ngx/paperless-ngx/pull/10751)) | ||||
| -   Chore(deps-dev): Bump @<!---->types/node from 24.1.0 to 24.3.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10750](https://github.com/paperless-ngx/paperless-ngx/pull/10750)) | ||||
| -   Chore(deps): Bump the actions group with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10757](https://github.com/paperless-ngx/paperless-ngx/pull/10757)) | ||||
| </details> | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>13 changes</summary> | ||||
|  | ||||
| -   Revert "Performance: Enable virtual scrolling for large custom field … @shamoon ([#10803](https://github.com/paperless-ngx/paperless-ngx/pull/10803)) | ||||
| -   Fixhancement: update sidebar view counts on save \& next also @shamoon ([#10793](https://github.com/paperless-ngx/paperless-ngx/pull/10793)) | ||||
| -   Enhancement: report websocket status @shamoon ([#10777](https://github.com/paperless-ngx/paperless-ngx/pull/10777)) | ||||
| -   Chore(deps-dev): Bump the frontend-jest-dependencies group in /src-ui with 2 updates @shamoon ([#10770](https://github.com/paperless-ngx/paperless-ngx/pull/10770)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10745](https://github.com/paperless-ngx/paperless-ngx/pull/10745)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 22 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10744](https://github.com/paperless-ngx/paperless-ngx/pull/10744)) | ||||
| -   Chore(deps): Bump bootstrap from 5.3.7 to 5.3.8 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10740](https://github.com/paperless-ngx/paperless-ngx/pull/10740)) | ||||
| -   Chore(deps-dev): Bump @<!---->playwright/test from 1.54.2 to 1.55.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10743](https://github.com/paperless-ngx/paperless-ngx/pull/10743)) | ||||
| -   Chore(deps-dev): Bump webpack from 5.101.0 to 5.101.3 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10751](https://github.com/paperless-ngx/paperless-ngx/pull/10751)) | ||||
| -   Chore(deps-dev): Bump @<!---->types/node from 24.1.0 to 24.3.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10750](https://github.com/paperless-ngx/paperless-ngx/pull/10750)) | ||||
| -   Chore: switch from os.path to pathlib.Path @gothicVI ([#10539](https://github.com/paperless-ngx/paperless-ngx/pull/10539)) | ||||
| -   Performance fix: add paging for custom field select options @shamoon ([#10755](https://github.com/paperless-ngx/paperless-ngx/pull/10755)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.18.3 | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: include application config language settings for dateparser auto-detection [@shamoon](https://github.com/shamoon) ([#10722](https://github.com/paperless-ngx/paperless-ngx/pull/10722)) | ||||
| -   Fix: hide sidebar counts during saved views organization [@shamoon](https://github.com/shamoon) ([#10716](https://github.com/paperless-ngx/paperless-ngx/pull/10716)) | ||||
| -   Fix: wrap long view titles in sidebar [@shamoon](https://github.com/shamoon) ([#10715](https://github.com/paperless-ngx/paperless-ngx/pull/10715)) | ||||
| -   Fixhancement: more saved view count refreshes [@shamoon](https://github.com/shamoon) ([#10694](https://github.com/paperless-ngx/paperless-ngx/pull/10694)) | ||||
| -   Fix: include pagination array items for valid openapi schema [@shamoon](https://github.com/shamoon) ([#10682](https://github.com/paperless-ngx/paperless-ngx/pull/10682)) | ||||
| -   Fix: prevent scroll for view name in sidebar [@shamoon](https://github.com/shamoon) ([#10676](https://github.com/paperless-ngx/paperless-ngx/pull/10676)) | ||||
| -   Tweak: center document close button in app frame [@shamoon](https://github.com/shamoon) ([#10661](https://github.com/paperless-ngx/paperless-ngx/pull/10661)) | ||||
| -   Performance: Enable virtual scrolling for large custom field selects @david-loe ([#10708](https://github.com/paperless-ngx/paperless-ngx/pull/10708)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| <details> | ||||
| <summary>5 changes</summary> | ||||
|  | ||||
| -   Chore(deps): Update granian[uvloop] requirement from ~=2.4.1 to ~=2.5.1 @[dependabot[bot]](https://github.com/apps/dependabot) ([#10529](https://github.com/paperless-ngx/paperless-ngx/pull/10529)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 6 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10714](https://github.com/paperless-ngx/paperless-ngx/pull/10714)) | ||||
| -   docker-compose(deps): Bump library/mariadb from 11 to 12 in /docker/compose @[dependabot[bot]](https://github.com/apps/dependabot) ([#10621](https://github.com/paperless-ngx/paperless-ngx/pull/10621)) | ||||
| -   docker-compose(deps): Bump gotenberg/gotenberg from 8.20 to 8.22 in /docker/compose @[dependabot[bot]](https://github.com/apps/dependabot) ([#10687](https://github.com/paperless-ngx/paperless-ngx/pull/10687)) | ||||
| -   docker(deps): Bump astral-sh/uv from 0.8.8-python3.12-bookworm-slim to 0.8.13-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10685](https://github.com/paperless-ngx/paperless-ngx/pull/10685)) | ||||
| </details> | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>11 changes</summary> | ||||
|  | ||||
| -   Fix: include application config language settings for dateparser auto-detection [@shamoon](https://github.com/shamoon) ([#10722](https://github.com/paperless-ngx/paperless-ngx/pull/10722)) | ||||
| -   Chore(deps): Update granian[uvloop] requirement from ~=2.4.1 to ~=2.5.1 @[dependabot[bot]](https://github.com/apps/dependabot) ([#10529](https://github.com/paperless-ngx/paperless-ngx/pull/10529)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 6 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10714](https://github.com/paperless-ngx/paperless-ngx/pull/10714)) | ||||
| -   Fix: hide sidebar counts during saved views organization [@shamoon](https://github.com/shamoon) ([#10716](https://github.com/paperless-ngx/paperless-ngx/pull/10716)) | ||||
| -   Fix: wrap long view titles in sidebar [@shamoon](https://github.com/shamoon) ([#10715](https://github.com/paperless-ngx/paperless-ngx/pull/10715)) | ||||
| -   Performance: Enable virtual scrolling for large custom field selects @david-loe ([#10708](https://github.com/paperless-ngx/paperless-ngx/pull/10708)) | ||||
| -   Chore: refactor document details component [@shamoon](https://github.com/shamoon) ([#10662](https://github.com/paperless-ngx/paperless-ngx/pull/10662)) | ||||
| -   Fixhancement: more saved view count refreshes [@shamoon](https://github.com/shamoon) ([#10694](https://github.com/paperless-ngx/paperless-ngx/pull/10694)) | ||||
| -   Fix: include pagination array items for valid openapi schema [@shamoon](https://github.com/shamoon) ([#10682](https://github.com/paperless-ngx/paperless-ngx/pull/10682)) | ||||
| -   Fix: prevent scroll for view name in sidebar [@shamoon](https://github.com/shamoon) ([#10676](https://github.com/paperless-ngx/paperless-ngx/pull/10676)) | ||||
| -   Tweak: center document close button in app frame [@shamoon](https://github.com/shamoon) ([#10661](https://github.com/paperless-ngx/paperless-ngx/pull/10661)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.18.2 | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: prevent loss of changes when switching between open docs [@shamoon](https://github.com/shamoon) ([#10659](https://github.com/paperless-ngx/paperless-ngx/pull/10659)) | ||||
| -   Fix: ignore incomplete tasks for system status 'last run' [@shamoon](https://github.com/shamoon) ([#10641](https://github.com/paperless-ngx/paperless-ngx/pull/10641)) | ||||
| -   Fix: increase legibility of date filter clear button in light mode [@shamoon](https://github.com/shamoon) ([#10649](https://github.com/paperless-ngx/paperless-ngx/pull/10649)) | ||||
| -   Fix: ensure saved view count is visible with long names [@shamoon](https://github.com/shamoon) ([#10616](https://github.com/paperless-ngx/paperless-ngx/pull/10616)) | ||||
| -   Tweak: improve dateparser auto-detection messages [@shamoon](https://github.com/shamoon) ([#10640](https://github.com/paperless-ngx/paperless-ngx/pull/10640)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| -   Chore(deps): Bump the development group across 1 directory with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10578](https://github.com/paperless-ngx/paperless-ngx/pull/10578)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>6 changes</summary> | ||||
|  | ||||
| -   Fix: prevent loss of changes when switching between open docs [@shamoon](https://github.com/shamoon) ([#10659](https://github.com/paperless-ngx/paperless-ngx/pull/10659)) | ||||
| -   Fix: ignore incomplete tasks for system status 'last run' [@shamoon](https://github.com/shamoon) ([#10641](https://github.com/paperless-ngx/paperless-ngx/pull/10641)) | ||||
| -   Tweak: improve dateparser auto-detection messages [@shamoon](https://github.com/shamoon) ([#10640](https://github.com/paperless-ngx/paperless-ngx/pull/10640)) | ||||
| -   Fix: increase legibility of date filter clear button in light mode [@shamoon](https://github.com/shamoon) ([#10649](https://github.com/paperless-ngx/paperless-ngx/pull/10649)) | ||||
| -   Fix: ensure saved view count is visible with long names [@shamoon](https://github.com/shamoon) ([#10616](https://github.com/paperless-ngx/paperless-ngx/pull/10616)) | ||||
| -   Chore(deps): Bump the development group across 1 directory with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10578](https://github.com/paperless-ngx/paperless-ngx/pull/10578)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.18.1 | ||||
|  | ||||
| ### Features / Enhancements | ||||
|  | ||||
| -   Tweak: fix some button consistency [@shamoon](https://github.com/shamoon) ([#10593](https://github.com/paperless-ngx/paperless-ngx/pull/10593)) | ||||
| -   Fixhancement: mobile layout improvements for pdf editor [@shamoon](https://github.com/shamoon) ([#10588](https://github.com/paperless-ngx/paperless-ngx/pull/10588)) | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: fix app logo validation with no file [@shamoon](https://github.com/shamoon) ([#10599](https://github.com/paperless-ngx/paperless-ngx/pull/10599)) | ||||
|  | ||||
| ### Documentation | ||||
|  | ||||
| -   Documentation: fix filters docs [@shamoon](https://github.com/shamoon) ([#10600](https://github.com/paperless-ngx/paperless-ngx/pull/10600)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>4 changes</summary> | ||||
|  | ||||
| -   Fix: fix app logo validation with no file [@shamoon](https://github.com/shamoon) ([#10599](https://github.com/paperless-ngx/paperless-ngx/pull/10599)) | ||||
| -   Tweak: fix some button consistency [@shamoon](https://github.com/shamoon) ([#10593](https://github.com/paperless-ngx/paperless-ngx/pull/10593)) | ||||
| -   Development: restore version tag display [@shamoon](https://github.com/shamoon) ([#10592](https://github.com/paperless-ngx/paperless-ngx/pull/10592)) | ||||
| -   Fixhancement: mobile layout improvements for pdf editor [@shamoon](https://github.com/shamoon) ([#10588](https://github.com/paperless-ngx/paperless-ngx/pull/10588)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.18.0 | ||||
|  | ||||
| ### Notable Changes | ||||
|  | ||||
| -   Feature: PDF editor [@shamoon](https://github.com/shamoon) ([#10318](https://github.com/paperless-ngx/paperless-ngx/pull/10318)) | ||||
|  | ||||
| ### Features / Enhancements | ||||
|  | ||||
| -   Feature: Add filter to localize dates for filepath templating [@stumpylog](https://github.com/stumpylog) ([#10559](https://github.com/paperless-ngx/paperless-ngx/pull/10559)) | ||||
| -   Feature: PDF editor [@shamoon](https://github.com/shamoon) ([#10318](https://github.com/paperless-ngx/paperless-ngx/pull/10318)) | ||||
| -   Enhancement: support webhook restrictions [@shamoon](https://github.com/shamoon) ([#10555](https://github.com/paperless-ngx/paperless-ngx/pull/10555)) | ||||
| -   Performance: Classifier performance optimizations [@Merinorus](https://github.com/Merinorus) ([#10363](https://github.com/paperless-ngx/paperless-ngx/pull/10363)) | ||||
| -   Performance: add setting to enable DB connection pooling for PostgreSQL [@Merinorus](https://github.com/Merinorus) ([#10354](https://github.com/paperless-ngx/paperless-ngx/pull/10354)) | ||||
| -   Fixhancement: improve text thumbnail generation for large files [@shamoon](https://github.com/shamoon) ([#10483](https://github.com/paperless-ngx/paperless-ngx/pull/10483)) | ||||
| -   Enhancement: disable auto spellcheck on filtering dropdowns [@TheDodger](https://github.com/TheDodger) ([#10487](https://github.com/paperless-ngx/paperless-ngx/pull/10487)) | ||||
| -   Enhancement: display saved view counts [@shamoon](https://github.com/shamoon) ([#10246](https://github.com/paperless-ngx/paperless-ngx/pull/10246)) | ||||
| -   Fixhancement: add missing exact operator for boolean CF queries [@shamoon](https://github.com/shamoon) ([#10402](https://github.com/paperless-ngx/paperless-ngx/pull/10402)) | ||||
| -   Feature: add Vietnamese translation [@shamoon](https://github.com/shamoon) ([#10352](https://github.com/paperless-ngx/paperless-ngx/pull/10352)) | ||||
| -   Performance: Add support for configuring date parser languages [@Merinorus](https://github.com/Merinorus) ([#10181](https://github.com/paperless-ngx/paperless-ngx/pull/10181)) | ||||
| -   Enhancement: Add a database caching for improved performance [@Merinorus](https://github.com/Merinorus) ([#9784](https://github.com/paperless-ngx/paperless-ngx/pull/9784)) | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: include ignore for config logos in sanity checker [@shamoon](https://github.com/shamoon) ([#10473](https://github.com/paperless-ngx/paperless-ngx/pull/10473)) | ||||
| -   Fix: track and restore changed document fields from session storage [@shamoon](https://github.com/shamoon) ([#10468](https://github.com/paperless-ngx/paperless-ngx/pull/10468)) | ||||
| -   Fix: Make some natural keyword date searches timezone-aware [@shamoon](https://github.com/shamoon) ([#10416](https://github.com/paperless-ngx/paperless-ngx/pull/10416)) | ||||
| -   Fixhancement: follow redirects in curl health check [@V0idC0de](https://github.com/V0idC0de) ([#10415](https://github.com/paperless-ngx/paperless-ngx/pull/10415)) | ||||
| -   Fix: dont use translated verbose_name for getting object perms [@shamoon](https://github.com/shamoon) ([#10399](https://github.com/paperless-ngx/paperless-ngx/pull/10399)) | ||||
| -   Fix: fix date format for 'today' in DateComponent [@shamoon](https://github.com/shamoon) ([#10369](https://github.com/paperless-ngx/paperless-ngx/pull/10369)) | ||||
| -   Fix: default to empty permissions for group creation [@shamoon](https://github.com/shamoon) ([#10337](https://github.com/paperless-ngx/paperless-ngx/pull/10337)) | ||||
| -   Fix: correct api created coercion with timezone [@shamoon](https://github.com/shamoon) ([#10287](https://github.com/paperless-ngx/paperless-ngx/pull/10287)) | ||||
| -   Fix: reset search query for preview on reset filter [@shamoon](https://github.com/shamoon) ([#10279](https://github.com/paperless-ngx/paperless-ngx/pull/10279)) | ||||
| -   Chore: reject absurd max age values [@shamoon](https://github.com/shamoon) ([#10243](https://github.com/paperless-ngx/paperless-ngx/pull/10243)) | ||||
| -   Chore: add tasks task_id param to openapi spec [@shamoon](https://github.com/shamoon) ([#10469](https://github.com/paperless-ngx/paperless-ngx/pull/10469)) | ||||
| -   Chore: include advanced search query param in API spec [@shamoon](https://github.com/shamoon) ([#10449](https://github.com/paperless-ngx/paperless-ngx/pull/10449)) | ||||
|  | ||||
| ### Security | ||||
|  | ||||
| -   Address XSS vulnerability GHSA-6p53-hqqw-8j62 | ||||
|  | ||||
| ### Maintenance | ||||
|  | ||||
| -   docker(deps): Bump astral-sh/uv from 0.8.4-python3.12-bookworm-slim to 0.8.8-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10564](https://github.com/paperless-ngx/paperless-ngx/pull/10564)) | ||||
| -   docker(deps): Bump astral-sh/uv from 0.7.9-python3.12-bookworm-slim to 0.7.19-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10343](https://github.com/paperless-ngx/paperless-ngx/pull/10343)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 7 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10347](https://github.com/paperless-ngx/paperless-ngx/pull/10347)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.15.29 to 24.0.10 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10306](https://github.com/paperless-ngx/paperless-ngx/pull/10306)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 8 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10481](https://github.com/paperless-ngx/paperless-ngx/pull/10481)) | ||||
| -   docker(deps): bump astral-sh/uv from 0.7.19-python3.12-bookworm-slim to 0.8.3-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10465](https://github.com/paperless-ngx/paperless-ngx/pull/10465)) | ||||
| -   Chore: switch from os.path to pathlib.Path [@gothicVI](https://github.com/gothicVI) ([#10397](https://github.com/paperless-ngx/paperless-ngx/pull/10397)) | ||||
| -   Chore(deps): Bump the small-changes group with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10528](https://github.com/paperless-ngx/paperless-ngx/pull/10528)) | ||||
| -   Chore(deps): Bump the django group across 1 directory with 9 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10538](https://github.com/paperless-ngx/paperless-ngx/pull/10538)) | ||||
| -   Chore(deps): Bump stefanzweifel/git-auto-commit-action from 5 to 6 in the actions group @[dependabot[bot]](https://github.com/apps/dependabot) ([#10302](https://github.com/paperless-ngx/paperless-ngx/pull/10302)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| <details> | ||||
| <summary>23 changes</summary> | ||||
|  | ||||
| -   chore: Small targeted upgrades to dependencies [@stumpylog](https://github.com/stumpylog) ([#10561](https://github.com/paperless-ngx/paperless-ngx/pull/10561)) | ||||
| -   docker(deps): Bump astral-sh/uv from 0.8.4-python3.12-bookworm-slim to 0.8.8-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10564](https://github.com/paperless-ngx/paperless-ngx/pull/10564)) | ||||
| -   Chore(deps): Bump the django group across 1 directory with 9 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10538](https://github.com/paperless-ngx/paperless-ngx/pull/10538)) | ||||
| -   Chore(deps): Bump the small-changes group with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10528](https://github.com/paperless-ngx/paperless-ngx/pull/10528)) | ||||
| -   Chore(deps-dev): Bump the frontend-jest-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10497](https://github.com/paperless-ngx/paperless-ngx/pull/10497)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10498](https://github.com/paperless-ngx/paperless-ngx/pull/10498)) | ||||
| -   Chore(deps-dev): Bump @playwright/test from 1.53.2 to 1.54.2 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10499](https://github.com/paperless-ngx/paperless-ngx/pull/10499)) | ||||
| -   Chore(deps-dev): Bump webpack from 5.99.9 to 5.101.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10501](https://github.com/paperless-ngx/paperless-ngx/pull/10501)) | ||||
| -   Chore(deps-dev): Bump prettier-plugin-organize-imports from 4.1.0 to 4.2.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10500](https://github.com/paperless-ngx/paperless-ngx/pull/10500)) | ||||
| -   Chore(deps-dev): Bump @types/node from 24.0.10 to 24.1.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10502](https://github.com/paperless-ngx/paperless-ngx/pull/10502)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 16 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10496](https://github.com/paperless-ngx/paperless-ngx/pull/10496)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 8 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10481](https://github.com/paperless-ngx/paperless-ngx/pull/10481)) | ||||
| -   docker(deps): bump astral-sh/uv from 0.7.19-python3.12-bookworm-slim to 0.8.3-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10465](https://github.com/paperless-ngx/paperless-ngx/pull/10465)) | ||||
| -   docker(deps): Bump astral-sh/uv from 0.7.9-python3.12-bookworm-slim to 0.7.19-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10343](https://github.com/paperless-ngx/paperless-ngx/pull/10343)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 7 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10347](https://github.com/paperless-ngx/paperless-ngx/pull/10347)) | ||||
| -   Chore(deps): Bump stefanzweifel/git-auto-commit-action from 5 to 6 in the actions group @[dependabot[bot]](https://github.com/apps/dependabot) ([#10302](https://github.com/paperless-ngx/paperless-ngx/pull/10302)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group across 1 directory with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10311](https://github.com/paperless-ngx/paperless-ngx/pull/10311)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.15.29 to 24.0.10 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10306](https://github.com/paperless-ngx/paperless-ngx/pull/10306)) | ||||
| -   Chore(deps): Bump bootstrap from 5.3.6 to 5.3.7 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10308](https://github.com/paperless-ngx/paperless-ngx/pull/10308)) | ||||
| -   Chore(deps-dev): Bump webpack from 5.98.0 to 5.99.9 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10309](https://github.com/paperless-ngx/paperless-ngx/pull/10309)) | ||||
| -   Chore(deps-dev): Bump @playwright/test from 1.51.1 to 1.53.2 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10307](https://github.com/paperless-ngx/paperless-ngx/pull/10307)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 13 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10303](https://github.com/paperless-ngx/paperless-ngx/pull/10303)) | ||||
| -   Chore: update to Angular 20 [@shamoon](https://github.com/shamoon) ([#10273](https://github.com/paperless-ngx/paperless-ngx/pull/10273)) | ||||
| </details> | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>44 changes</summary> | ||||
|  | ||||
| -   chore: Small targeted upgrades to dependencies [@stumpylog](https://github.com/stumpylog) ([#10561](https://github.com/paperless-ngx/paperless-ngx/pull/10561)) | ||||
| -   Feature: Add filter to localize dates for filepath templating [@stumpylog](https://github.com/stumpylog) ([#10559](https://github.com/paperless-ngx/paperless-ngx/pull/10559)) | ||||
| -   Chore: Removes duplication and spread out config for codespell [@stumpylog](https://github.com/stumpylog) ([#10560](https://github.com/paperless-ngx/paperless-ngx/pull/10560)) | ||||
| -   Chore(deps): Bump the django group across 1 directory with 9 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10538](https://github.com/paperless-ngx/paperless-ngx/pull/10538)) | ||||
| -   Feature: PDF editor [@shamoon](https://github.com/shamoon) ([#10318](https://github.com/paperless-ngx/paperless-ngx/pull/10318)) | ||||
| -   Enhancement: support webhook restrictions [@shamoon](https://github.com/shamoon) ([#10555](https://github.com/paperless-ngx/paperless-ngx/pull/10555)) | ||||
| -   Performance: Classifier performance optimizations [@Merinorus](https://github.com/Merinorus) ([#10363](https://github.com/paperless-ngx/paperless-ngx/pull/10363)) | ||||
| -   Chore: switch from os.path to pathlib.Path [@gothicVI](https://github.com/gothicVI) ([#10397](https://github.com/paperless-ngx/paperless-ngx/pull/10397)) | ||||
| -   Chore(deps): Bump the small-changes group with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10528](https://github.com/paperless-ngx/paperless-ngx/pull/10528)) | ||||
| -   Performance: add setting to enable DB connection pooling for PostgreSQL [@Merinorus](https://github.com/Merinorus) ([#10354](https://github.com/paperless-ngx/paperless-ngx/pull/10354)) | ||||
| -   Chore(deps-dev): Bump the frontend-jest-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10497](https://github.com/paperless-ngx/paperless-ngx/pull/10497)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10498](https://github.com/paperless-ngx/paperless-ngx/pull/10498)) | ||||
| -   Chore(deps-dev): Bump @playwright/test from 1.53.2 to 1.54.2 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10499](https://github.com/paperless-ngx/paperless-ngx/pull/10499)) | ||||
| -   Chore(deps-dev): Bump webpack from 5.99.9 to 5.101.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10501](https://github.com/paperless-ngx/paperless-ngx/pull/10501)) | ||||
| -   Chore(deps-dev): Bump prettier-plugin-organize-imports from 4.1.0 to 4.2.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10500](https://github.com/paperless-ngx/paperless-ngx/pull/10500)) | ||||
| -   Chore(deps-dev): Bump @types/node from 24.0.10 to 24.1.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10502](https://github.com/paperless-ngx/paperless-ngx/pull/10502)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 16 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10496](https://github.com/paperless-ngx/paperless-ngx/pull/10496)) | ||||
| -   Fixhancement: improve text thumbnail generation for large files [@shamoon](https://github.com/shamoon) ([#10483](https://github.com/paperless-ngx/paperless-ngx/pull/10483)) | ||||
| -   Enhancement: disable auto spellcheck on filtering dropdowns [@TheDodger](https://github.com/TheDodger) ([#10487](https://github.com/paperless-ngx/paperless-ngx/pull/10487)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 8 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10481](https://github.com/paperless-ngx/paperless-ngx/pull/10481)) | ||||
| -   Fix: include ignore for config logos in sanity checker [@shamoon](https://github.com/shamoon) ([#10473](https://github.com/paperless-ngx/paperless-ngx/pull/10473)) | ||||
| -   Chore: add tasks task_id param to openapi spec [@shamoon](https://github.com/shamoon) ([#10469](https://github.com/paperless-ngx/paperless-ngx/pull/10469)) | ||||
| -   Fix: track and restore changed document fields from session storage [@shamoon](https://github.com/shamoon) ([#10468](https://github.com/paperless-ngx/paperless-ngx/pull/10468)) | ||||
| -   Chore: include advanced search query param in API spec [@shamoon](https://github.com/shamoon) ([#10449](https://github.com/paperless-ngx/paperless-ngx/pull/10449)) | ||||
| -   Enhancement: display saved view counts [@shamoon](https://github.com/shamoon) ([#10246](https://github.com/paperless-ngx/paperless-ngx/pull/10246)) | ||||
| -   Fix: Make some natural keyword date searches timezone-aware [@shamoon](https://github.com/shamoon) ([#10416](https://github.com/paperless-ngx/paperless-ngx/pull/10416)) | ||||
| -   Fixhancement: add missing exact operator for boolean CF queries [@shamoon](https://github.com/shamoon) ([#10402](https://github.com/paperless-ngx/paperless-ngx/pull/10402)) | ||||
| -   Fix: dont use translated verbose_name for getting object perms [@shamoon](https://github.com/shamoon) ([#10399](https://github.com/paperless-ngx/paperless-ngx/pull/10399)) | ||||
| -   Fix: fix date format for 'today' in DateComponent [@shamoon](https://github.com/shamoon) ([#10369](https://github.com/paperless-ngx/paperless-ngx/pull/10369)) | ||||
| -   Feature: add Vietnamese translation [@shamoon](https://github.com/shamoon) ([#10352](https://github.com/paperless-ngx/paperless-ngx/pull/10352)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 7 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10347](https://github.com/paperless-ngx/paperless-ngx/pull/10347)) | ||||
| -   Fix: default to empty permissions for group creation [@shamoon](https://github.com/shamoon) ([#10337](https://github.com/paperless-ngx/paperless-ngx/pull/10337)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group across 1 directory with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10311](https://github.com/paperless-ngx/paperless-ngx/pull/10311)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.15.29 to 24.0.10 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10306](https://github.com/paperless-ngx/paperless-ngx/pull/10306)) | ||||
| -   Chore(deps): Bump bootstrap from 5.3.6 to 5.3.7 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10308](https://github.com/paperless-ngx/paperless-ngx/pull/10308)) | ||||
| -   Chore(deps-dev): Bump webpack from 5.98.0 to 5.99.9 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10309](https://github.com/paperless-ngx/paperless-ngx/pull/10309)) | ||||
| -   Chore(deps-dev): Bump @playwright/test from 1.51.1 to 1.53.2 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10307](https://github.com/paperless-ngx/paperless-ngx/pull/10307)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 13 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10303](https://github.com/paperless-ngx/paperless-ngx/pull/10303)) | ||||
| -   Performance: Add support for configuring date parser languages [@Merinorus](https://github.com/Merinorus) ([#10181](https://github.com/paperless-ngx/paperless-ngx/pull/10181)) | ||||
| -   Enhancement: Add a database caching for improved performance [@Merinorus](https://github.com/Merinorus) ([#9784](https://github.com/paperless-ngx/paperless-ngx/pull/9784)) | ||||
| -   Fix: correct api created coercion with timezone [@shamoon](https://github.com/shamoon) ([#10287](https://github.com/paperless-ngx/paperless-ngx/pull/10287)) | ||||
| -   Fix: reset search query for preview on reset filter [@shamoon](https://github.com/shamoon) ([#10279](https://github.com/paperless-ngx/paperless-ngx/pull/10279)) | ||||
| -   Chore: update to Angular 20 [@shamoon](https://github.com/shamoon) ([#10273](https://github.com/paperless-ngx/paperless-ngx/pull/10273)) | ||||
| -   Chore: reject absurd max age values [@shamoon](https://github.com/shamoon) ([#10243](https://github.com/paperless-ngx/paperless-ngx/pull/10243)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.17.1 | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: correct PAPERLESS_EMPTY_TRASH_DIR to Path [@shamoon](https://github.com/shamoon) ([#10227](https://github.com/paperless-ngx/paperless-ngx/pull/10227)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| -   Fix: correct PAPERLESS_EMPTY_TRASH_DIR to Path [@shamoon](https://github.com/shamoon) ([#10227](https://github.com/paperless-ngx/paperless-ngx/pull/10227)) | ||||
|  | ||||
| ## paperless-ngx 2.17.0 | ||||
|  | ||||
| ### Breaking Changes | ||||
|  | ||||
| -   Fix: restore expected pre-2.16 scheduled workflow offset behavior [@shamoon](https://github.com/shamoon) ([#10218](https://github.com/paperless-ngx/paperless-ngx/pull/10218)) | ||||
|  | ||||
| ### Features / Enhancements | ||||
|  | ||||
| -   QoL: log version at startup, show backend vs frontend mismatch in system status [@shamoon](https://github.com/shamoon) ([#10214](https://github.com/paperless-ngx/paperless-ngx/pull/10214)) | ||||
| -   Feature: add Persian translation [@shamoon](https://github.com/shamoon) ([#10183](https://github.com/paperless-ngx/paperless-ngx/pull/10183)) | ||||
| -   Enhancement: support import of zipped export [@kaerbr](https://github.com/kaerbr) ([#10073](https://github.com/paperless-ngx/paperless-ngx/pull/10073)) | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: more api fixes [@shamoon](https://github.com/shamoon) ([#10204](https://github.com/paperless-ngx/paperless-ngx/pull/10204)) | ||||
| -   Fix: restore expected pre-2.16 scheduled workflow offset behavior [@shamoon](https://github.com/shamoon) ([#10218](https://github.com/paperless-ngx/paperless-ngx/pull/10218)) | ||||
| -   Fix: fix some API crashes [@shamoon](https://github.com/shamoon) ([#10196](https://github.com/paperless-ngx/paperless-ngx/pull/10196)) | ||||
| -   Fix: remove duplicate base path in websocket urls [@robertmx](https://github.com/robertmx) ([#10194](https://github.com/paperless-ngx/paperless-ngx/pull/10194)) | ||||
| -   Fix: use hard delete for custom fields with workflow removal [@shamoon](https://github.com/shamoon) ([#10191](https://github.com/paperless-ngx/paperless-ngx/pull/10191)) | ||||
| -   Fix: fix mail account test api schema [@shamoon](https://github.com/shamoon) ([#10164](https://github.com/paperless-ngx/paperless-ngx/pull/10164)) | ||||
| -   Fix: correct api schema for mail_account process [@shamoon](https://github.com/shamoon) ([#10157](https://github.com/paperless-ngx/paperless-ngx/pull/10157)) | ||||
| -   Fix: correct api schema for next_asn [@shamoon](https://github.com/shamoon) ([#10151](https://github.com/paperless-ngx/paperless-ngx/pull/10151)) | ||||
| -   Fix: fix email and notes endpoints api spec [@shamoon](https://github.com/shamoon) ([#10148](https://github.com/paperless-ngx/paperless-ngx/pull/10148)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| -   Chore: bump angular/common to 19.12.14 [@shamoon](https://github.com/shamoon) ([#10212](https://github.com/paperless-ngx/paperless-ngx/pull/10212)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>14 changes</summary> | ||||
|  | ||||
| -   QoL: log version at startup, show backend vs frontend mismatch in system status [@shamoon](https://github.com/shamoon) ([#10214](https://github.com/paperless-ngx/paperless-ngx/pull/10214)) | ||||
| -   Fix: more api fixes [@shamoon](https://github.com/shamoon) ([#10204](https://github.com/paperless-ngx/paperless-ngx/pull/10204)) | ||||
| -   Fix: restore expected pre-2.16 scheduled workflow offset behavior [@shamoon](https://github.com/shamoon) ([#10218](https://github.com/paperless-ngx/paperless-ngx/pull/10218)) | ||||
| -   Chore: switch from os.path to pathlib.Path [@gothicVI](https://github.com/gothicVI) ([#9933](https://github.com/paperless-ngx/paperless-ngx/pull/9933)) | ||||
| -   Chore: bump angular/common to 19.12.14 [@shamoon](https://github.com/shamoon) ([#10212](https://github.com/paperless-ngx/paperless-ngx/pull/10212)) | ||||
| -   Fix: fix some API crashes [@shamoon](https://github.com/shamoon) ([#10196](https://github.com/paperless-ngx/paperless-ngx/pull/10196)) | ||||
| -   Fix: remove duplicate base path in websocket urls [@robertmx](https://github.com/robertmx) ([#10194](https://github.com/paperless-ngx/paperless-ngx/pull/10194)) | ||||
| -   Fix: use hard delete for custom fields with workflow removal [@shamoon](https://github.com/shamoon) ([#10191](https://github.com/paperless-ngx/paperless-ngx/pull/10191)) | ||||
| -   Feature: add Persian translation [@shamoon](https://github.com/shamoon) ([#10183](https://github.com/paperless-ngx/paperless-ngx/pull/10183)) | ||||
| -   Enhancement: support import of zipped export [@kaerbr](https://github.com/kaerbr) ([#10073](https://github.com/paperless-ngx/paperless-ngx/pull/10073)) | ||||
| -   Fix: fix mail account test api schema [@shamoon](https://github.com/shamoon) ([#10164](https://github.com/paperless-ngx/paperless-ngx/pull/10164)) | ||||
| -   Fix: correct api schema for mail_account process [@shamoon](https://github.com/shamoon) ([#10157](https://github.com/paperless-ngx/paperless-ngx/pull/10157)) | ||||
| -   Fix: correct api schema for next_asn [@shamoon](https://github.com/shamoon) ([#10151](https://github.com/paperless-ngx/paperless-ngx/pull/10151)) | ||||
| -   Fix: fix email and notes endpoints api spec [@shamoon](https://github.com/shamoon) ([#10148](https://github.com/paperless-ngx/paperless-ngx/pull/10148)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.16.3 | ||||
|  | ||||
| ### Features / Enhancements | ||||
|  | ||||
| -   Performance: pre-filter document list in scheduled workflow checks [@shamoon](https://github.com/shamoon) ([#10031](https://github.com/paperless-ngx/paperless-ngx/pull/10031)) | ||||
| -   Chore: refactor consumer plugin checks to a pre-flight plugin [@shamoon](https://github.com/shamoon) ([#9994](https://github.com/paperless-ngx/paperless-ngx/pull/9994)) | ||||
| -   Enhancement: include DOCUMENT_TYPE to post consume scripts [@matthesrieke](https://github.com/matthesrieke) ([#9977](https://github.com/paperless-ngx/paperless-ngx/pull/9977)) | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: handle whoosh query correction errors [@shamoon](https://github.com/shamoon) ([#10121](https://github.com/paperless-ngx/paperless-ngx/pull/10121)) | ||||
| -   Fix: handle favicon with non-default static dir [@shamoon](https://github.com/shamoon) ([#10107](https://github.com/paperless-ngx/paperless-ngx/pull/10107)) | ||||
| -   Fixhancement: cleanup user or group references in settings upon deletion [@shamoon](https://github.com/shamoon) ([#10049](https://github.com/paperless-ngx/paperless-ngx/pull/10049)) | ||||
| -   Fix: Add exception to in [@Freilichtbuehne](https://github.com/Freilichtbuehne) ([#10070](https://github.com/paperless-ngx/paperless-ngx/pull/10070)) | ||||
| -   Fix: include base href when opening global search result in new window [@shamoon](https://github.com/shamoon) ([#10066](https://github.com/paperless-ngx/paperless-ngx/pull/10066)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| <details> | ||||
| <summary>10 changes</summary> | ||||
|  | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10085](https://github.com/paperless-ngx/paperless-ngx/pull/10085)) | ||||
| -   Chore(deps): Update granian[uvloop] requirement from ~=2.2.0 to ~=2.3.2 @[dependabot[bot]](https://github.com/apps/dependabot) ([#10055](https://github.com/paperless-ngx/paperless-ngx/pull/10055)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 18 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10099](https://github.com/paperless-ngx/paperless-ngx/pull/10099)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10100](https://github.com/paperless-ngx/paperless-ngx/pull/10100)) | ||||
| -   Chore(deps): Bump bootstrap from 5.3.3 to 5.3.6 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10091](https://github.com/paperless-ngx/paperless-ngx/pull/10091)) | ||||
| -   Chore(deps-dev): Bump @codecov/webpack-plugin from 1.9.0 to 1.9.1 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10090](https://github.com/paperless-ngx/paperless-ngx/pull/10090)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.15.3 to 22.15.29 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10089](https://github.com/paperless-ngx/paperless-ngx/pull/10089)) | ||||
| -   Chore(deps): Bump zone.js from 0.15.0 to 0.15.1 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10088](https://github.com/paperless-ngx/paperless-ngx/pull/10088)) | ||||
| -   docker(deps): bump astral-sh/uv from 0.7.8-python3.12-bookworm-slim to 0.7.9-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10084](https://github.com/paperless-ngx/paperless-ngx/pull/10084)) | ||||
| -   docker(deps): Bump astral-sh/uv from 0.6.16-python3.12-bookworm-slim to 0.7.8-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#10056](https://github.com/paperless-ngx/paperless-ngx/pull/10056)) | ||||
| </details> | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>16 changes</summary> | ||||
|  | ||||
| -   Fix: handle whoosh query correction errors [@shamoon](https://github.com/shamoon) ([#10121](https://github.com/paperless-ngx/paperless-ngx/pull/10121)) | ||||
| -   Performance: pre-filter document list in scheduled workflow checks [@shamoon](https://github.com/shamoon) ([#10031](https://github.com/paperless-ngx/paperless-ngx/pull/10031)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10085](https://github.com/paperless-ngx/paperless-ngx/pull/10085)) | ||||
| -   Chore: refactor consumer plugin checks to a pre-flight plugin [@shamoon](https://github.com/shamoon) ([#9994](https://github.com/paperless-ngx/paperless-ngx/pull/9994)) | ||||
| -   Chore(deps): Update granian[uvloop] requirement from ~=2.2.0 to ~=2.3.2 @[dependabot[bot]](https://github.com/apps/dependabot) ([#10055](https://github.com/paperless-ngx/paperless-ngx/pull/10055)) | ||||
| -   Fix: handle favicon with non-default static dir [@shamoon](https://github.com/shamoon) ([#10107](https://github.com/paperless-ngx/paperless-ngx/pull/10107)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 18 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10099](https://github.com/paperless-ngx/paperless-ngx/pull/10099)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#10100](https://github.com/paperless-ngx/paperless-ngx/pull/10100)) | ||||
| -   Chore(deps): Bump bootstrap from 5.3.3 to 5.3.6 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10091](https://github.com/paperless-ngx/paperless-ngx/pull/10091)) | ||||
| -   Chore(deps-dev): Bump @codecov/webpack-plugin from 1.9.0 to 1.9.1 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10090](https://github.com/paperless-ngx/paperless-ngx/pull/10090)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.15.3 to 22.15.29 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10089](https://github.com/paperless-ngx/paperless-ngx/pull/10089)) | ||||
| -   Chore(deps): Bump zone.js from 0.15.0 to 0.15.1 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#10088](https://github.com/paperless-ngx/paperless-ngx/pull/10088)) | ||||
| -   Fixhancement: cleanup user or group references in settings upon deletion [@shamoon](https://github.com/shamoon) ([#10049](https://github.com/paperless-ngx/paperless-ngx/pull/10049)) | ||||
| -   Enhancement: include DOCUMENT_TYPE to post consume scripts [@matthesrieke](https://github.com/matthesrieke) ([#9977](https://github.com/paperless-ngx/paperless-ngx/pull/9977)) | ||||
| -   Fix: Add exception to in [@Freilichtbuehne](https://github.com/Freilichtbuehne) ([#10070](https://github.com/paperless-ngx/paperless-ngx/pull/10070)) | ||||
| -   Fix: include base href when opening global search result in new window [@shamoon](https://github.com/shamoon) ([#10066](https://github.com/paperless-ngx/paperless-ngx/pull/10066)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.16.2 | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: accept datetime for created [@shamoon](https://github.com/shamoon) ([#10021](https://github.com/paperless-ngx/paperless-ngx/pull/10021)) | ||||
| -   Fix: created date fixes in v2.16 [@shamoon](https://github.com/shamoon) ([#10026](https://github.com/paperless-ngx/paperless-ngx/pull/10026)) | ||||
| -   Fix: mark fields for created objects as dirty [@shamoon](https://github.com/shamoon) ([#10022](https://github.com/paperless-ngx/paperless-ngx/pull/10022)) | ||||
| -   Fix: add fallback to copyfile on PermissionError @samuel-kosmann ([#10023](https://github.com/paperless-ngx/paperless-ngx/pull/10023)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| -   Chore: warn users about removal of postgres v13 support [@shamoon](https://github.com/shamoon) ([#9980](https://github.com/paperless-ngx/paperless-ngx/pull/9980)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>5 changes</summary> | ||||
|  | ||||
| -   Fix: accept datetime for created [@shamoon](https://github.com/shamoon) ([#10021](https://github.com/paperless-ngx/paperless-ngx/pull/10021)) | ||||
| -   Fix: add fallback to copyfile on PermissionError @samuel-kosmann ([#10023](https://github.com/paperless-ngx/paperless-ngx/pull/10023)) | ||||
| -   Fix: created date fixes in v2.16 [@shamoon](https://github.com/shamoon) ([#10026](https://github.com/paperless-ngx/paperless-ngx/pull/10026)) | ||||
| -   Fix: mark fields for created objects as dirty [@shamoon](https://github.com/shamoon) ([#10022](https://github.com/paperless-ngx/paperless-ngx/pull/10022)) | ||||
| -   Chore: warn users about removal of postgres v13 support [@shamoon](https://github.com/shamoon) ([#9980](https://github.com/paperless-ngx/paperless-ngx/pull/9980)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.16.1 | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: fix created date filtering broken in 2.16.0 [@shamoon](https://github.com/shamoon) ([#9976](https://github.com/paperless-ngx/paperless-ngx/pull/9976)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| -   Fix: fix created date filtering broken in 2.16.0 [@shamoon](https://github.com/shamoon) ([#9976](https://github.com/paperless-ngx/paperless-ngx/pull/9976)) | ||||
|  | ||||
| ## paperless-ngx 2.16.0 | ||||
|  | ||||
| ### Breaking Changes | ||||
|  | ||||
| -   [BREAKING] Change: treat created as date not datetime [@shamoon](https://github.com/shamoon) ([#9793](https://github.com/paperless-ngx/paperless-ngx/pull/9793)) | ||||
|  | ||||
| ### Features | ||||
|  | ||||
| -   Enhancement: support negative offset in scheduled workflows [@shamoon](https://github.com/shamoon) ([#9746](https://github.com/paperless-ngx/paperless-ngx/pull/9746)) | ||||
| -   Enhancement: support heic images [@shamoon](https://github.com/shamoon) ([#9771](https://github.com/paperless-ngx/paperless-ngx/pull/9771)) | ||||
| -   Enhancement: use patch instead of put for frontend document changes [@shamoon](https://github.com/shamoon) ([#9744](https://github.com/paperless-ngx/paperless-ngx/pull/9744)) | ||||
| -   Fixhancement: automatically disable email verification if no smtp setup [@shamoon](https://github.com/shamoon) ([#9949](https://github.com/paperless-ngx/paperless-ngx/pull/9949)) | ||||
| -   Fixhancement: better handle removed social apps in profile [@shamoon](https://github.com/shamoon) ([#9876](https://github.com/paperless-ngx/paperless-ngx/pull/9876)) | ||||
| -   Enhancement: add barcode frontend config [@shamoon](https://github.com/shamoon) ([#9742](https://github.com/paperless-ngx/paperless-ngx/pull/9742)) | ||||
| -   Enhancement: support allauth disable unknown account emails [@shamoon](https://github.com/shamoon) ([#9743](https://github.com/paperless-ngx/paperless-ngx/pull/9743)) | ||||
| -   Fixhancement: tag plus button should add tag to doc [@shamoon](https://github.com/shamoon) ([#9762](https://github.com/paperless-ngx/paperless-ngx/pull/9762)) | ||||
| -   Fixhancement: check more permissions for status consumer messages [@shamoon](https://github.com/shamoon) ([#9804](https://github.com/paperless-ngx/paperless-ngx/pull/9804)) | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: include subpath in drf-spectacular settings if set [@shamoon](https://github.com/shamoon) ([#9738](https://github.com/paperless-ngx/paperless-ngx/pull/9738)) | ||||
| -   Fix: handle created change with api version increment, use created only on frontend, deprecate created_date [@shamoon](https://github.com/shamoon) ([#9962](https://github.com/paperless-ngx/paperless-ngx/pull/9962)) | ||||
| -   Fix: ignore logo file from sanity checker [@shamoon](https://github.com/shamoon) ([#9946](https://github.com/paperless-ngx/paperless-ngx/pull/9946)) | ||||
| -   Fix: correctly handle empty user for old notes api format, fix frontend API version [@shamoon](https://github.com/shamoon) ([#9846](https://github.com/paperless-ngx/paperless-ngx/pull/9846)) | ||||
| -   Fix: fix single select in filterable dropdowns when editing [@shamoon](https://github.com/shamoon) ([#9834](https://github.com/paperless-ngx/paperless-ngx/pull/9834)) | ||||
| -   Fix: always update classifier task result [@shamoon](https://github.com/shamoon) ([#9817](https://github.com/paperless-ngx/paperless-ngx/pull/9817)) | ||||
| -   Fix: fix zoom increase/decrease buttons in FF [@shamoon](https://github.com/shamoon) ([#9761](https://github.com/paperless-ngx/paperless-ngx/pull/9761)) | ||||
|  | ||||
| ### Maintenance | ||||
|  | ||||
| -   Chore(deps): Bump astral-sh/setup-uv from 5 to 6 in the actions group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9842](https://github.com/paperless-ngx/paperless-ngx/pull/9842)) | ||||
| -   Chore: split ci frontend e2e vs unit tests [@shamoon](https://github.com/shamoon) ([#9851](https://github.com/paperless-ngx/paperless-ngx/pull/9851)) | ||||
| -   Chore: auto-generate translation strings [@shamoon](https://github.com/shamoon) ([#9462](https://github.com/paperless-ngx/paperless-ngx/pull/9462)) | ||||
| -   Chore: add ymlfmt [@shamoon](https://github.com/shamoon) ([#9745](https://github.com/paperless-ngx/paperless-ngx/pull/9745)) | ||||
| -   Chore: replace secretary with GHA [@shamoon](https://github.com/shamoon) ([#9723](https://github.com/paperless-ngx/paperless-ngx/pull/9723)) | ||||
| -   Chore: resolve dynamic import warnings from pdfjs, again [@shamoon](https://github.com/shamoon) ([#9924](https://github.com/paperless-ngx/paperless-ngx/pull/9924)) | ||||
| -   Fix/Chore: replace file drop package [@shamoon](https://github.com/shamoon) ([#9926](https://github.com/paperless-ngx/paperless-ngx/pull/9926)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| <details> | ||||
| <summary>14 changes</summary> | ||||
|  | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 6 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9921](https://github.com/paperless-ngx/paperless-ngx/pull/9921)) | ||||
| -   docker-compose(deps): Bump library/redis from 7 to 8 in /docker/compose @[dependabot[bot]](https://github.com/apps/dependabot) ([#9879](https://github.com/paperless-ngx/paperless-ngx/pull/9879)) | ||||
| -   Chore(deps): Bump astral-sh/setup-uv from 5 to 6 in the actions group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9842](https://github.com/paperless-ngx/paperless-ngx/pull/9842)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 14 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9848](https://github.com/paperless-ngx/paperless-ngx/pull/9848)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9849](https://github.com/paperless-ngx/paperless-ngx/pull/9849)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.17 to 22.15.3 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9850](https://github.com/paperless-ngx/paperless-ngx/pull/9850)) | ||||
| -   docker(deps): Bump astral-sh/uv from 0.6.14-python3.12-bookworm-slim to 0.6.16-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#9767](https://github.com/paperless-ngx/paperless-ngx/pull/9767)) | ||||
| -   docker-compose(deps): bump gotenberg/gotenberg from 8.19 to 8.20 in /docker/compose @[dependabot[bot]](https://github.com/apps/dependabot) ([#9661](https://github.com/paperless-ngx/paperless-ngx/pull/9661)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 17 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9768](https://github.com/paperless-ngx/paperless-ngx/pull/9768)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9770](https://github.com/paperless-ngx/paperless-ngx/pull/9770)) | ||||
| -   Chore(deps-dev): Bump jest-preset-angular from 14.5.4 to 14.5.5 in /src-ui in the frontend-jest-dependencies group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9769](https://github.com/paperless-ngx/paperless-ngx/pull/9769)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 6 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9764](https://github.com/paperless-ngx/paperless-ngx/pull/9764)) | ||||
| -   Chore(deps): Bump the django group across 1 directory with 6 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9753](https://github.com/paperless-ngx/paperless-ngx/pull/9753)) | ||||
| -   docker(deps): bump astral-sh/uv from 0.6.13-python3.12-bookworm-slim to 0.6.14-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#9656](https://github.com/paperless-ngx/paperless-ngx/pull/9656)) | ||||
| </details> | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>29 changes</summary> | ||||
|  | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 6 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9921](https://github.com/paperless-ngx/paperless-ngx/pull/9921)) | ||||
| -   Fix: handle created change with api version increment, use created only on frontend, deprecate created_date [@shamoon](https://github.com/shamoon) ([#9962](https://github.com/paperless-ngx/paperless-ngx/pull/9962)) | ||||
| -   Fixhancement: automatically disable email verification if no smtp setup [@shamoon](https://github.com/shamoon) ([#9949](https://github.com/paperless-ngx/paperless-ngx/pull/9949)) | ||||
| -   Fix: ignore logo file from sanity checker [@shamoon](https://github.com/shamoon) ([#9946](https://github.com/paperless-ngx/paperless-ngx/pull/9946)) | ||||
| -   [BREAKING] Change: treat created as date not datetime [@shamoon](https://github.com/shamoon) ([#9793](https://github.com/paperless-ngx/paperless-ngx/pull/9793)) | ||||
| -   Fix/Chore: replace file drop package [@shamoon](https://github.com/shamoon) ([#9926](https://github.com/paperless-ngx/paperless-ngx/pull/9926)) | ||||
| -   Chore: resolve dynamic import warnings from pdfjs, again [@shamoon](https://github.com/shamoon) ([#9924](https://github.com/paperless-ngx/paperless-ngx/pull/9924)) | ||||
| -   Enhancement: support negative offset in scheduled workflows [@shamoon](https://github.com/shamoon) ([#9746](https://github.com/paperless-ngx/paperless-ngx/pull/9746)) | ||||
| -   Fixhancement: better handle removed social apps in profile [@shamoon](https://github.com/shamoon) ([#9876](https://github.com/paperless-ngx/paperless-ngx/pull/9876)) | ||||
| -   Enhancement: add barcode frontend config [@shamoon](https://github.com/shamoon) ([#9742](https://github.com/paperless-ngx/paperless-ngx/pull/9742)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 14 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9848](https://github.com/paperless-ngx/paperless-ngx/pull/9848)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9849](https://github.com/paperless-ngx/paperless-ngx/pull/9849)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.17 to 22.15.3 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9850](https://github.com/paperless-ngx/paperless-ngx/pull/9850)) | ||||
| -   Fix: correctly handle empty user for old notes api format, fix frontend API version [@shamoon](https://github.com/shamoon) ([#9846](https://github.com/paperless-ngx/paperless-ngx/pull/9846)) | ||||
| -   Fix: fix single select in filterable dropdowns when editing [@shamoon](https://github.com/shamoon) ([#9834](https://github.com/paperless-ngx/paperless-ngx/pull/9834)) | ||||
| -   Fix: always update classifier task result [@shamoon](https://github.com/shamoon) ([#9817](https://github.com/paperless-ngx/paperless-ngx/pull/9817)) | ||||
| -   Fixhancement: check more permissions for status consumer messages [@shamoon](https://github.com/shamoon) ([#9804](https://github.com/paperless-ngx/paperless-ngx/pull/9804)) | ||||
| -   Enhancement: support heic images [@shamoon](https://github.com/shamoon) ([#9771](https://github.com/paperless-ngx/paperless-ngx/pull/9771)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 17 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9768](https://github.com/paperless-ngx/paperless-ngx/pull/9768)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9770](https://github.com/paperless-ngx/paperless-ngx/pull/9770)) | ||||
| -   Chore(deps-dev): Bump jest-preset-angular from 14.5.4 to 14.5.5 in /src-ui in the frontend-jest-dependencies group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9769](https://github.com/paperless-ngx/paperless-ngx/pull/9769)) | ||||
| -   Enhancement: support allauth disable unknown account emails [@shamoon](https://github.com/shamoon) ([#9743](https://github.com/paperless-ngx/paperless-ngx/pull/9743)) | ||||
| -   Enhancement: use patch instead of put for frontend document changes [@shamoon](https://github.com/shamoon) ([#9744](https://github.com/paperless-ngx/paperless-ngx/pull/9744)) | ||||
| -   Chore(deps): Bump the small-changes group across 1 directory with 6 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9764](https://github.com/paperless-ngx/paperless-ngx/pull/9764)) | ||||
| -   Chore(deps): Bump the django group across 1 directory with 6 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9753](https://github.com/paperless-ngx/paperless-ngx/pull/9753)) | ||||
| -   Fixhancement: tag plus button should add tag to doc [@shamoon](https://github.com/shamoon) ([#9762](https://github.com/paperless-ngx/paperless-ngx/pull/9762)) | ||||
| -   Fix: fix zoom increase/decrease buttons in FF [@shamoon](https://github.com/shamoon) ([#9761](https://github.com/paperless-ngx/paperless-ngx/pull/9761)) | ||||
| -   Chore: switch from os.path to pathlib.Path [@gothicVI](https://github.com/gothicVI) ([#9339](https://github.com/paperless-ngx/paperless-ngx/pull/9339)) | ||||
| -   Fix: include subpath in drf-spectacular settings if set [@shamoon](https://github.com/shamoon) ([#9738](https://github.com/paperless-ngx/paperless-ngx/pull/9738)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.15.3 | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: do not try deleting original file that was moved to trash dir [@shamoon](https://github.com/shamoon) ([#9684](https://github.com/paperless-ngx/paperless-ngx/pull/9684)) | ||||
| -   Fix: preserve non-ASCII filenames in document downloads [@shamoon](https://github.com/shamoon) ([#9702](https://github.com/paperless-ngx/paperless-ngx/pull/9702)) | ||||
| -   Fix: fix breaking api change to document notes user field [@shamoon](https://github.com/shamoon) ([#9714](https://github.com/paperless-ngx/paperless-ngx/pull/9714)) | ||||
| -   Fix: another doc link fix [@shamoon](https://github.com/shamoon) ([#9700](https://github.com/paperless-ngx/paperless-ngx/pull/9700)) | ||||
| -   Fix: correctly handle dict data with webhook [@shamoon](https://github.com/shamoon) ([#9674](https://github.com/paperless-ngx/paperless-ngx/pull/9674)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>5 changes</summary> | ||||
|  | ||||
| -   Fix: do not try deleting original file that was moved to trash dir [@shamoon](https://github.com/shamoon) ([#9684](https://github.com/paperless-ngx/paperless-ngx/pull/9684)) | ||||
| -   Fix: preserve non-ASCII filenames in document downloads [@shamoon](https://github.com/shamoon) ([#9702](https://github.com/paperless-ngx/paperless-ngx/pull/9702)) | ||||
| -   Fix: fix breaking api change to document notes user field [@shamoon](https://github.com/shamoon) ([#9714](https://github.com/paperless-ngx/paperless-ngx/pull/9714)) | ||||
| -   Fix: another doc link fix [@shamoon](https://github.com/shamoon) ([#9700](https://github.com/paperless-ngx/paperless-ngx/pull/9700)) | ||||
| -   Fix: correctly handle dict data with webhook [@shamoon](https://github.com/shamoon) ([#9674](https://github.com/paperless-ngx/paperless-ngx/pull/9674)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.15.2 | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: Adds better handling during folder checking/creation/permissions for non-root [@stumpylog](https://github.com/stumpylog) ([#9616](https://github.com/paperless-ngx/paperless-ngx/pull/9616)) | ||||
| -   Fix: Explicitly set the HOME environment to resolve issues running as root with database certificates [@stumpylog](https://github.com/stumpylog) ([#9643](https://github.com/paperless-ngx/paperless-ngx/pull/9643)) | ||||
| -   Fix: prevent self-linking when bulk edit doc link [@shamoon](https://github.com/shamoon) ([#9629](https://github.com/paperless-ngx/paperless-ngx/pull/9629)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| -   Chore: Bump celery to 5.5.1 [@hannesortmeier](https://github.com/hannesortmeier) ([#9642](https://github.com/paperless-ngx/paperless-ngx/pull/9642)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>4 changes</summary> | ||||
|  | ||||
| -   Tweak: consistently use created date when displaying doc in list [@shamoon](https://github.com/shamoon) ([#9651](https://github.com/paperless-ngx/paperless-ngx/pull/9651)) | ||||
| -   Fix: Adds better handling during folder checking/creation/permissions for non-root [@stumpylog](https://github.com/stumpylog) ([#9616](https://github.com/paperless-ngx/paperless-ngx/pull/9616)) | ||||
| -   Fix: Explicitly set the HOME environment to resolve issues running as root with database certificates [@stumpylog](https://github.com/stumpylog) ([#9643](https://github.com/paperless-ngx/paperless-ngx/pull/9643)) | ||||
| -   Fix: prevent self-linking when bulk edit doc link [@shamoon](https://github.com/shamoon) ([#9629](https://github.com/paperless-ngx/paperless-ngx/pull/9629)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.15.1 | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: Run migration lock as the correct user [@stumpylog](https://github.com/stumpylog) ([#9604](https://github.com/paperless-ngx/paperless-ngx/pull/9604)) | ||||
| -   Fix: Adds a warning to the user if their secret file includes a trailing newline [@stumpylog](https://github.com/stumpylog) ([#9601](https://github.com/paperless-ngx/paperless-ngx/pull/9601)) | ||||
| -   Fix: correct download filename in 2.15.0 [@shamoon](https://github.com/shamoon) ([#9599](https://github.com/paperless-ngx/paperless-ngx/pull/9599)) | ||||
| -   Fix: dont exclude matching check for scheduled workflows [@shamoon](https://github.com/shamoon) ([#9594](https://github.com/paperless-ngx/paperless-ngx/pull/9594)) | ||||
|  | ||||
| ### Maintenance | ||||
|  | ||||
| -   docker(deps): Bump astral-sh/uv from 0.6.9-python3.12-bookworm-slim to 0.6.13-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#9573](https://github.com/paperless-ngx/paperless-ngx/pull/9573)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| -   docker(deps): Bump astral-sh/uv from 0.6.9-python3.12-bookworm-slim to 0.6.13-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#9573](https://github.com/paperless-ngx/paperless-ngx/pull/9573)) | ||||
| -   Chore: move to whoosh-reloaded, for now [@shamoon](https://github.com/shamoon) ([#9605](https://github.com/paperless-ngx/paperless-ngx/pull/9605)) | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>4 changes</summary> | ||||
|  | ||||
| -   Fix: Run migration lock as the correct user [@stumpylog](https://github.com/stumpylog) ([#9604](https://github.com/paperless-ngx/paperless-ngx/pull/9604)) | ||||
| -   Fix: Adds a warning to the user if their secret file includes a trailing newline [@stumpylog](https://github.com/stumpylog) ([#9601](https://github.com/paperless-ngx/paperless-ngx/pull/9601)) | ||||
| -   Fix: correct download filename in 2.15.0 [@shamoon](https://github.com/shamoon) ([#9599](https://github.com/paperless-ngx/paperless-ngx/pull/9599)) | ||||
| -   Fix: dont exclude matching check for scheduled workflows [@shamoon](https://github.com/shamoon) ([#9594](https://github.com/paperless-ngx/paperless-ngx/pull/9594)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.15.0 | ||||
|  | ||||
| ### Features | ||||
|  | ||||
| -   Enhancement: allow webUI first account signup [@shamoon](https://github.com/shamoon) ([#9500](https://github.com/paperless-ngx/paperless-ngx/pull/9500)) | ||||
| -   Enhancement: support more 'not assigned' filtering, refactor [@shamoon](https://github.com/shamoon) ([#9429](https://github.com/paperless-ngx/paperless-ngx/pull/9429)) | ||||
| -   Enhancement: reorganize dates dropdown, add more relative options [@shamoon](https://github.com/shamoon) ([#9307](https://github.com/paperless-ngx/paperless-ngx/pull/9307)) | ||||
| -   Enhancement: add switch to allow merging non-PDFs with archive version [@shamoon](https://github.com/shamoon) ([#9305](https://github.com/paperless-ngx/paperless-ngx/pull/9305)) | ||||
| -   Enhancement: support assigning custom field values in workflows [@shamoon](https://github.com/shamoon) ([#9272](https://github.com/paperless-ngx/paperless-ngx/pull/9272)) | ||||
| -   Enhancement: Add slugify filter in templating [@hwaterke](https://github.com/hwaterke) ([#9269](https://github.com/paperless-ngx/paperless-ngx/pull/9269)) | ||||
| -   Feature: Switch webserver to granian [@stumpylog](https://github.com/stumpylog) ([#9218](https://github.com/paperless-ngx/paperless-ngx/pull/9218)) | ||||
| -   Enhancement: relocate and smaller upload widget, dont limit upload list [@shamoon](https://github.com/shamoon) ([#9244](https://github.com/paperless-ngx/paperless-ngx/pull/9244)) | ||||
| -   Enhancement: run tasks from system status, report sanity check, simpler classifier check, styling updates [@shamoon](https://github.com/shamoon) ([#9106](https://github.com/paperless-ngx/paperless-ngx/pull/9106)) | ||||
| -   Enhancement: include celery log in logs view [@shamoon](https://github.com/shamoon) ([#9214](https://github.com/paperless-ngx/paperless-ngx/pull/9214)) | ||||
| -   Enhancement: support default groups for regular and social account signup, syncing on login [@shamoon](https://github.com/shamoon) ([#9039](https://github.com/paperless-ngx/paperless-ngx/pull/9039)) | ||||
| -   Enhancement: allow disabling the filesystem consumer [@shamoon](https://github.com/shamoon) ([#9199](https://github.com/paperless-ngx/paperless-ngx/pull/9199)) | ||||
| -   Feature: email document [@shamoon](https://github.com/shamoon) ([#8950](https://github.com/paperless-ngx/paperless-ngx/pull/8950)) | ||||
| -   Enhancement: webui workflowtrigger source option [@shamoon](https://github.com/shamoon) ([#9170](https://github.com/paperless-ngx/paperless-ngx/pull/9170)) | ||||
| -   Enhancement: use charfield for webhook url, custom validation [@shamoon](https://github.com/shamoon) ([#9128](https://github.com/paperless-ngx/paperless-ngx/pull/9128)) | ||||
| -   Feature: Chinese Traditional translation [@LokiHung](https://github.com/LokiHung) ([#9076](https://github.com/paperless-ngx/paperless-ngx/pull/9076)) | ||||
| -   Enhancement: Use cached sessions for a minor performance improvement [@stumpylog](https://github.com/stumpylog) ([#9074](https://github.com/paperless-ngx/paperless-ngx/pull/9074)) | ||||
| -   Feature: openapi spec, full api browser [@shamoon](https://github.com/shamoon) ([#8948](https://github.com/paperless-ngx/paperless-ngx/pull/8948)) | ||||
| -   Enhancement: filter by file type [@shamoon](https://github.com/shamoon) ([#8946](https://github.com/paperless-ngx/paperless-ngx/pull/8946)) | ||||
| -   Feature: Transition Docker to use s6 overlay [@stumpylog](https://github.com/stumpylog) ([#8886](https://github.com/paperless-ngx/paperless-ngx/pull/8886)) | ||||
| -   Feature: better toast notifications management [@shamoon](https://github.com/shamoon) ([#8980](https://github.com/paperless-ngx/paperless-ngx/pull/8980)) | ||||
| -   Enhancement: date picker and date filter dropdown improvements [@shamoon](https://github.com/shamoon) ([#9033](https://github.com/paperless-ngx/paperless-ngx/pull/9033)) | ||||
| -   Tweak: more accurate classifier last trained time [@shamoon](https://github.com/shamoon) ([#9004](https://github.com/paperless-ngx/paperless-ngx/pull/9004)) | ||||
| -   Enhancement: allow setting default pdf zoom [@shamoon](https://github.com/shamoon) ([#9017](https://github.com/paperless-ngx/paperless-ngx/pull/9017)) | ||||
|  | ||||
| ### Bug Fixes | ||||
|  | ||||
| -   Fix: ensure only matched scheduled workflows are applied [@shamoon](https://github.com/shamoon) ([#9580](https://github.com/paperless-ngx/paperless-ngx/pull/9580)) | ||||
| -   Fix: fix large doc thumb hidden at unexpected screen sizes [@shamoon](https://github.com/shamoon) ([#9559](https://github.com/paperless-ngx/paperless-ngx/pull/9559)) | ||||
| -   Fix: fix potential race condition when creating new cf from doc details [@shamoon](https://github.com/shamoon) ([#9542](https://github.com/paperless-ngx/paperless-ngx/pull/9542)) | ||||
| -   Fix: fix doc link input [@shamoon](https://github.com/shamoon) ([#9533](https://github.com/paperless-ngx/paperless-ngx/pull/9533)) | ||||
| -   Fix: only overwrite existing cf values in workflow if set [@shamoon](https://github.com/shamoon) ([#9459](https://github.com/paperless-ngx/paperless-ngx/pull/9459)) | ||||
| -   Fix: fix auto-close when doc update no longer has permissions [@shamoon](https://github.com/shamoon) ([#9453](https://github.com/paperless-ngx/paperless-ngx/pull/9453)) | ||||
| -   Change: better handle permissions in patch requests [@shamoon](https://github.com/shamoon) ([#9393](https://github.com/paperless-ngx/paperless-ngx/pull/9393)) | ||||
| -   Fix: use correct filename with webhook [@shamoon](https://github.com/shamoon) ([#9392](https://github.com/paperless-ngx/paperless-ngx/pull/9392)) | ||||
| -   Change: sync OIDC groups on first login too [@shamoon](https://github.com/shamoon) ([#9387](https://github.com/paperless-ngx/paperless-ngx/pull/9387)) | ||||
| -   Fix: only parse custom field queries when valid [@shamoon](https://github.com/shamoon) ([#9384](https://github.com/paperless-ngx/paperless-ngx/pull/9384)) | ||||
| -   Fix: Allow setting of other Granian options [@stumpylog](https://github.com/stumpylog) ([#9360](https://github.com/paperless-ngx/paperless-ngx/pull/9360)) | ||||
| -   Fix: Always clean up INotify [@stumpylog](https://github.com/stumpylog) ([#9359](https://github.com/paperless-ngx/paperless-ngx/pull/9359)) | ||||
| -   Fix typo in inactive account template [@ocean90](https://github.com/ocean90) ([#9356](https://github.com/paperless-ngx/paperless-ngx/pull/9356)) | ||||
| -   Fix: fix notes serializing in API document response [@shamoon](https://github.com/shamoon) ([#9336](https://github.com/paperless-ngx/paperless-ngx/pull/9336)) | ||||
| -   Fix: correct all results with whoosh queries [@shamoon](https://github.com/shamoon) ([#9331](https://github.com/paperless-ngx/paperless-ngx/pull/9331)) | ||||
| -   Fix: fix typo in altered migration [@gothicVI](https://github.com/gothicVI) ([#9321](https://github.com/paperless-ngx/paperless-ngx/pull/9321)) | ||||
| -   Fix: add account_inactive template / url [@shamoon](https://github.com/shamoon) ([#9322](https://github.com/paperless-ngx/paperless-ngx/pull/9322)) | ||||
| -   Fix: Switches data to content to upload raw bytes/text content [@stumpylog](https://github.com/stumpylog) ([#9293](https://github.com/paperless-ngx/paperless-ngx/pull/9293)) | ||||
| -   Fix: handle null workflow body and email subject [@shamoon](https://github.com/shamoon) ([#9271](https://github.com/paperless-ngx/paperless-ngx/pull/9271)) | ||||
| -   Fix: cleanup saved view references on custom field deletion, auto-refresh views, show error on saved view save [@shamoon](https://github.com/shamoon) ([#9225](https://github.com/paperless-ngx/paperless-ngx/pull/9225)) | ||||
| -   Fix: revert thumbnail CSS workaround in favor of GPU workaround [@shamoon](https://github.com/shamoon) ([#9219](https://github.com/paperless-ngx/paperless-ngx/pull/9219)) | ||||
| -   Fix: correct split confirm removal [@shamoon](https://github.com/shamoon) ([#9195](https://github.com/paperless-ngx/paperless-ngx/pull/9195)) | ||||
| -   Fix: saved views do not return to default display fields after setting and then removing [@shamoon](https://github.com/shamoon) ([#9168](https://github.com/paperless-ngx/paperless-ngx/pull/9168)) | ||||
| -   Fix: correct logged number of deleted documents on trash empty [@shamoon](https://github.com/shamoon) ([#9148](https://github.com/paperless-ngx/paperless-ngx/pull/9148)) | ||||
| -   Fix: include account confirm email allauth URL [@shamoon](https://github.com/shamoon) ([#9147](https://github.com/paperless-ngx/paperless-ngx/pull/9147)) | ||||
| -   Fix: remove additional scrollbar from popup preview [@shamoon](https://github.com/shamoon) ([#9140](https://github.com/paperless-ngx/paperless-ngx/pull/9140)) | ||||
| -   Fix: wrap selected display fields [@shamoon](https://github.com/shamoon) ([#9139](https://github.com/paperless-ngx/paperless-ngx/pull/9139)) | ||||
| -   Fix: reset documents sort field if user deletes the custom field [@shamoon](https://github.com/shamoon) ([#9127](https://github.com/paperless-ngx/paperless-ngx/pull/9127)) | ||||
| -   Fix: limit document title length in workflows [@shamoon](https://github.com/shamoon) ([#9085](https://github.com/paperless-ngx/paperless-ngx/pull/9085)) | ||||
| -   Fix: include doc link input import in custom fields query dropdown [@shamoon](https://github.com/shamoon) ([#9058](https://github.com/paperless-ngx/paperless-ngx/pull/9058)) | ||||
| -   Fix: deselect and trigger refresh for deleted documents from bulk operations with delete originals [@shamoon](https://github.com/shamoon) ([#8996](https://github.com/paperless-ngx/paperless-ngx/pull/8996)) | ||||
| -   Fix: allow empty email in profile [@shamoon](https://github.com/shamoon) ([#9012](https://github.com/paperless-ngx/paperless-ngx/pull/9012)) | ||||
|  | ||||
| ### Maintenance | ||||
|  | ||||
| -   docker(deps): Bump astral-sh/uv from 0.6.5-python3.12-bookworm-slim to 0.6.9-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#9488](https://github.com/paperless-ngx/paperless-ngx/pull/9488)) | ||||
| -   Chore: Enables dependabot for Dockerfile and our Compose files [@stumpylog](https://github.com/stumpylog) ([#9342](https://github.com/paperless-ngx/paperless-ngx/pull/9342)) | ||||
| -   Chore: ensure codecov upload gets attempted [@shamoon](https://github.com/shamoon) ([#9308](https://github.com/paperless-ngx/paperless-ngx/pull/9308)) | ||||
| -   Chore: Split out some items into extras [@stumpylog](https://github.com/stumpylog) ([#9297](https://github.com/paperless-ngx/paperless-ngx/pull/9297)) | ||||
| -   Chore: Enables Codecov test reporting for the backend [@stumpylog](https://github.com/stumpylog) ([#9295](https://github.com/paperless-ngx/paperless-ngx/pull/9295)) | ||||
| -   Chore: Combine Python settings files [@stumpylog](https://github.com/stumpylog) ([#9292](https://github.com/paperless-ngx/paperless-ngx/pull/9292)) | ||||
|  | ||||
| ### Dependencies | ||||
|  | ||||
| <details> | ||||
| <summary>43 changes</summary> | ||||
|  | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 20 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9536](https://github.com/paperless-ngx/paperless-ngx/pull/9536)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9538](https://github.com/paperless-ngx/paperless-ngx/pull/9538)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.9 to 22.13.17 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9539](https://github.com/paperless-ngx/paperless-ngx/pull/9539)) | ||||
| -   Chore(deps-dev): Bump jest-preset-angular from 14.5.3 to 14.5.4 in /src-ui in the frontend-jest-dependencies group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9537](https://github.com/paperless-ngx/paperless-ngx/pull/9537)) | ||||
| -   Chore(deps-dev): Bump @playwright/test from 1.50.1 to 1.51.1 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9540](https://github.com/paperless-ngx/paperless-ngx/pull/9540)) | ||||
| -   Chore(deps): Bump django from 5.1.6 to 5.1.7 in the django group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9486](https://github.com/paperless-ngx/paperless-ngx/pull/9486)) | ||||
| -   docker(deps): Bump astral-sh/uv from 0.6.5-python3.12-bookworm-slim to 0.6.9-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#9488](https://github.com/paperless-ngx/paperless-ngx/pull/9488)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9372](https://github.com/paperless-ngx/paperless-ngx/pull/9372)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 20 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9371](https://github.com/paperless-ngx/paperless-ngx/pull/9371)) | ||||
| -   Chore(deps): Update ocrmypdf requirement from ~=16.9.0 to ~=16.10.0 @[dependabot[bot]](https://github.com/apps/dependabot) ([#9348](https://github.com/paperless-ngx/paperless-ngx/pull/9348)) | ||||
| -   Chore(deps): Update drf-spectacular-sidecar requirement from ~=2025.2.1 to ~=2025.3.1 @[dependabot[bot]](https://github.com/apps/dependabot) ([#9347](https://github.com/paperless-ngx/paperless-ngx/pull/9347)) | ||||
| -   Chore(deps): Bump the small-changes group with 2 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9345](https://github.com/paperless-ngx/paperless-ngx/pull/9345)) | ||||
| -   docker-compose(deps): Bump library/postgres from 16 to 17 in /docker/compose @[dependabot[bot]](https://github.com/apps/dependabot) ([#9353](https://github.com/paperless-ngx/paperless-ngx/pull/9353)) | ||||
| -   docker(deps): Bump astral-sh/uv from 0.6.3-python3.12-bookworm-slim to 0.6.5-python3.12-bookworm-slim @[dependabot[bot]](https://github.com/apps/dependabot) ([#9344](https://github.com/paperless-ngx/paperless-ngx/pull/9344)) | ||||
| -   Chore(deps-dev): Bump the frontend-angular-dependencies group in /src-ui with 5 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9288](https://github.com/paperless-ngx/paperless-ngx/pull/9288)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.8 to 22.13.9 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9290](https://github.com/paperless-ngx/paperless-ngx/pull/9290)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9289](https://github.com/paperless-ngx/paperless-ngx/pull/9289)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.5 to 22.13.8 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9267](https://github.com/paperless-ngx/paperless-ngx/pull/9267)) | ||||
| -   Chore(deps): Bump stumpylog/image-cleaner-action from 0.9.0 to 0.10.0 in the actions group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9252](https://github.com/paperless-ngx/paperless-ngx/pull/9252)) | ||||
| -   Chore(deps-dev): Bump the development group with 2 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9253](https://github.com/paperless-ngx/paperless-ngx/pull/9253)) | ||||
| -   Chore(deps-dev): Bump @codecov/webpack-plugin from 1.8.0 to 1.9.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9260](https://github.com/paperless-ngx/paperless-ngx/pull/9260)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9256](https://github.com/paperless-ngx/paperless-ngx/pull/9256)) | ||||
| -   Chore(deps): Bump uuid from 11.0.5 to 11.1.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9259](https://github.com/paperless-ngx/paperless-ngx/pull/9259)) | ||||
| -   Chore(deps-dev): Bump jest-preset-angular from 14.5.1 to 14.5.3 in /src-ui in the frontend-jest-dependencies group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9255](https://github.com/paperless-ngx/paperless-ngx/pull/9255)) | ||||
| -   Chore(deps): Bump rxjs from 7.8.1 to 7.8.2 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9258](https://github.com/paperless-ngx/paperless-ngx/pull/9258)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.0 to 22.13.5 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9257](https://github.com/paperless-ngx/paperless-ngx/pull/9257)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 22 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9254](https://github.com/paperless-ngx/paperless-ngx/pull/9254)) | ||||
| -   Chore(deps): Bump django-filter from 24.3 to 25.1 in the django group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9143](https://github.com/paperless-ngx/paperless-ngx/pull/9143)) | ||||
| -   Chore(deps-dev): Bump mkdocs-material from 9.6.3 to 9.6.4 in the development group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9142](https://github.com/paperless-ngx/paperless-ngx/pull/9142)) | ||||
| -   Dependencies: Updates to jbig2enc 0.30 [@stumpylog](https://github.com/stumpylog) ([#9092](https://github.com/paperless-ngx/paperless-ngx/pull/9092)) | ||||
| -   Chore(deps): Bump cryptography from 44.0.0 to 44.0.1 @[dependabot[bot]](https://github.com/apps/dependabot) ([#9080](https://github.com/paperless-ngx/paperless-ngx/pull/9080)) | ||||
| -   Chore(deps): Bump the small-changes group with 7 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9064](https://github.com/paperless-ngx/paperless-ngx/pull/9064)) | ||||
| -   Chore(deps-dev): Bump the development group with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9061](https://github.com/paperless-ngx/paperless-ngx/pull/9061)) | ||||
| -   Chore(deps): Bump the django group across 1 directory with 2 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9065](https://github.com/paperless-ngx/paperless-ngx/pull/9065)) | ||||
| -   Chore(deps): Bump drf-spectacular-sidecar from 2024.11.1 to 2025.2.1 in the major-versions group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9063](https://github.com/paperless-ngx/paperless-ngx/pull/9063)) | ||||
| -   Chore(deps-dev): Bump the development group with 2 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9013](https://github.com/paperless-ngx/paperless-ngx/pull/9013)) | ||||
| -   Chore(deps): Bump django-soft-delete from 1.0.16 to 1.0.18 in the django group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9014](https://github.com/paperless-ngx/paperless-ngx/pull/9014)) | ||||
| -   Chore(deps): Bump uuid from 11.0.2 to 11.0.5 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#8992](https://github.com/paperless-ngx/paperless-ngx/pull/8992)) | ||||
| -   Chore(deps-dev): Bump @codecov/webpack-plugin from 1.2.1 to 1.8.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#8991](https://github.com/paperless-ngx/paperless-ngx/pull/8991)) | ||||
| -   Chore(deps-dev): Bump @playwright/test from 1.48.2 to 1.50.1 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#8993](https://github.com/paperless-ngx/paperless-ngx/pull/8993)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.8.6 to 22.13.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#8989](https://github.com/paperless-ngx/paperless-ngx/pull/8989)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#8988](https://github.com/paperless-ngx/paperless-ngx/pull/8988)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 23 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#8986](https://github.com/paperless-ngx/paperless-ngx/pull/8986)) | ||||
| </details> | ||||
|  | ||||
| ### All App Changes | ||||
|  | ||||
| <details> | ||||
| <summary>109 changes</summary> | ||||
|  | ||||
| -   Fix: ensure only matched scheduled workflows are applied [@shamoon](https://github.com/shamoon) ([#9580](https://github.com/paperless-ngx/paperless-ngx/pull/9580)) | ||||
| -   Fix: fix large doc thumb hidden at unexpected screen sizes [@shamoon](https://github.com/shamoon) ([#9559](https://github.com/paperless-ngx/paperless-ngx/pull/9559)) | ||||
| -   Fix: fix potential race condition when creating new cf from doc details [@shamoon](https://github.com/shamoon) ([#9542](https://github.com/paperless-ngx/paperless-ngx/pull/9542)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 20 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9536](https://github.com/paperless-ngx/paperless-ngx/pull/9536)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9538](https://github.com/paperless-ngx/paperless-ngx/pull/9538)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.9 to 22.13.17 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9539](https://github.com/paperless-ngx/paperless-ngx/pull/9539)) | ||||
| -   Chore(deps-dev): Bump jest-preset-angular from 14.5.3 to 14.5.4 in /src-ui in the frontend-jest-dependencies group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9537](https://github.com/paperless-ngx/paperless-ngx/pull/9537)) | ||||
| -   Chore(deps-dev): Bump @playwright/test from 1.50.1 to 1.51.1 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9540](https://github.com/paperless-ngx/paperless-ngx/pull/9540)) | ||||
| -   Fix: fix doc link input [@shamoon](https://github.com/shamoon) ([#9533](https://github.com/paperless-ngx/paperless-ngx/pull/9533)) | ||||
| -   Enhancement: allow webUI first account signup [@shamoon](https://github.com/shamoon) ([#9500](https://github.com/paperless-ngx/paperless-ngx/pull/9500)) | ||||
| -   Fix: fix cf dropdown placement on mobile [@shamoon](https://github.com/shamoon) ([#9508](https://github.com/paperless-ngx/paperless-ngx/pull/9508)) | ||||
| -   Chore(deps): Bump django from 5.1.6 to 5.1.7 in the django group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9486](https://github.com/paperless-ngx/paperless-ngx/pull/9486)) | ||||
| -   Fix: only overwrite existing cf values in workflow if set [@shamoon](https://github.com/shamoon) ([#9459](https://github.com/paperless-ngx/paperless-ngx/pull/9459)) | ||||
| -   Fix: fix auto-close when doc update no longer has permissions [@shamoon](https://github.com/shamoon) ([#9453](https://github.com/paperless-ngx/paperless-ngx/pull/9453)) | ||||
| -   Enhancement: support more 'not assigned' filtering, refactor [@shamoon](https://github.com/shamoon) ([#9429](https://github.com/paperless-ngx/paperless-ngx/pull/9429)) | ||||
| -   Change: better handle permissions in patch requests [@shamoon](https://github.com/shamoon) ([#9393](https://github.com/paperless-ngx/paperless-ngx/pull/9393)) | ||||
| -   Fix: use correct filename with webhook [@shamoon](https://github.com/shamoon) ([#9392](https://github.com/paperless-ngx/paperless-ngx/pull/9392)) | ||||
| -   Change: sync OIDC groups on first login too [@shamoon](https://github.com/shamoon) ([#9387](https://github.com/paperless-ngx/paperless-ngx/pull/9387)) | ||||
| -   Fix: only parse custom field queries when valid [@shamoon](https://github.com/shamoon) ([#9384](https://github.com/paperless-ngx/paperless-ngx/pull/9384)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9372](https://github.com/paperless-ngx/paperless-ngx/pull/9372)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 20 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9371](https://github.com/paperless-ngx/paperless-ngx/pull/9371)) | ||||
| -   Development: change frontend package manager to pnpm [@shamoon](https://github.com/shamoon) ([#9363](https://github.com/paperless-ngx/paperless-ngx/pull/9363)) | ||||
| -   Fix: Allow setting of other Granian options [@stumpylog](https://github.com/stumpylog) ([#9360](https://github.com/paperless-ngx/paperless-ngx/pull/9360)) | ||||
| -   Fix: Always clean up INotify [@stumpylog](https://github.com/stumpylog) ([#9359](https://github.com/paperless-ngx/paperless-ngx/pull/9359)) | ||||
| -   Tweak: add saved views hint to dashboard [@shamoon](https://github.com/shamoon) ([#9362](https://github.com/paperless-ngx/paperless-ngx/pull/9362)) | ||||
| -   Chore(deps): Update ocrmypdf requirement from ~=16.9.0 to ~=16.10.0 @[dependabot[bot]](https://github.com/apps/dependabot) ([#9348](https://github.com/paperless-ngx/paperless-ngx/pull/9348)) | ||||
| -   Chore(deps): Update drf-spectacular-sidecar requirement from ~=2025.2.1 to ~=2025.3.1 @[dependabot[bot]](https://github.com/apps/dependabot) ([#9347](https://github.com/paperless-ngx/paperless-ngx/pull/9347)) | ||||
| -   Chore(deps): Bump the small-changes group with 2 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9345](https://github.com/paperless-ngx/paperless-ngx/pull/9345)) | ||||
| -   Ensure the directories have been overridden and created for this test [@stumpylog](https://github.com/stumpylog) ([#9354](https://github.com/paperless-ngx/paperless-ngx/pull/9354)) | ||||
| -   Fix typo in inactive account template [@ocean90](https://github.com/ocean90) ([#9356](https://github.com/paperless-ngx/paperless-ngx/pull/9356)) | ||||
| -   Fix: fix notes serializing in API document response [@shamoon](https://github.com/shamoon) ([#9336](https://github.com/paperless-ngx/paperless-ngx/pull/9336)) | ||||
| -   Fix: correct all results with whoosh queries [@shamoon](https://github.com/shamoon) ([#9331](https://github.com/paperless-ngx/paperless-ngx/pull/9331)) | ||||
| -   Fix: fix typo in altered migration [@gothicVI](https://github.com/gothicVI) ([#9321](https://github.com/paperless-ngx/paperless-ngx/pull/9321)) | ||||
| -   Fix: add account_inactive template / url [@shamoon](https://github.com/shamoon) ([#9322](https://github.com/paperless-ngx/paperless-ngx/pull/9322)) | ||||
| -   Chore: Switch from os.path to pathlib.Path [@gothicVI](https://github.com/gothicVI) ([#9060](https://github.com/paperless-ngx/paperless-ngx/pull/9060)) | ||||
| -   Enhancement: reorganize dates dropdown, add more relative options [@shamoon](https://github.com/shamoon) ([#9307](https://github.com/paperless-ngx/paperless-ngx/pull/9307)) | ||||
| -   Chore: remove popper preventOverflow fix [@shamoon](https://github.com/shamoon) ([#9306](https://github.com/paperless-ngx/paperless-ngx/pull/9306)) | ||||
| -   Enhancement: add switch to allow merging non-PDFs with archive version [@shamoon](https://github.com/shamoon) ([#9305](https://github.com/paperless-ngx/paperless-ngx/pull/9305)) | ||||
| -   Enhancement: support assigning custom field values in workflows [@shamoon](https://github.com/shamoon) ([#9272](https://github.com/paperless-ngx/paperless-ngx/pull/9272)) | ||||
| -   Chore: add codecov frontend test results [@shamoon](https://github.com/shamoon) ([#9296](https://github.com/paperless-ngx/paperless-ngx/pull/9296)) | ||||
| -   Chore: Removes undocumented FileInfo [@stumpylog](https://github.com/stumpylog) ([#9298](https://github.com/paperless-ngx/paperless-ngx/pull/9298)) | ||||
| -   Chore(deps-dev): Bump the frontend-angular-dependencies group in /src-ui with 5 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9288](https://github.com/paperless-ngx/paperless-ngx/pull/9288)) | ||||
| -   Fix: Switches data to content to upload raw bytes/text content [@stumpylog](https://github.com/stumpylog) ([#9293](https://github.com/paperless-ngx/paperless-ngx/pull/9293)) | ||||
| -   Chore: Removes the unused Log model and LogFilterSet [@stumpylog](https://github.com/stumpylog) ([#9294](https://github.com/paperless-ngx/paperless-ngx/pull/9294)) | ||||
| -   Chore: Combine Python settings files [@stumpylog](https://github.com/stumpylog) ([#9292](https://github.com/paperless-ngx/paperless-ngx/pull/9292)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.8 to 22.13.9 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9290](https://github.com/paperless-ngx/paperless-ngx/pull/9290)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9289](https://github.com/paperless-ngx/paperless-ngx/pull/9289)) | ||||
| -   Chore: Switch from pipenv to uv [@stumpylog](https://github.com/stumpylog) ([#9251](https://github.com/paperless-ngx/paperless-ngx/pull/9251)) | ||||
| -   Enhancement: Add slugify filter in templating [@hwaterke](https://github.com/hwaterke) ([#9269](https://github.com/paperless-ngx/paperless-ngx/pull/9269)) | ||||
| -   Fix: handle null workflow body and email subject [@shamoon](https://github.com/shamoon) ([#9271](https://github.com/paperless-ngx/paperless-ngx/pull/9271)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.5 to 22.13.8 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9267](https://github.com/paperless-ngx/paperless-ngx/pull/9267)) | ||||
| -   Chore(deps-dev): Bump the development group with 2 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9253](https://github.com/paperless-ngx/paperless-ngx/pull/9253)) | ||||
| -   Chore(deps-dev): Bump @codecov/webpack-plugin from 1.8.0 to 1.9.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9260](https://github.com/paperless-ngx/paperless-ngx/pull/9260)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9256](https://github.com/paperless-ngx/paperless-ngx/pull/9256)) | ||||
| -   Chore(deps): Bump uuid from 11.0.5 to 11.1.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9259](https://github.com/paperless-ngx/paperless-ngx/pull/9259)) | ||||
| -   Chore(deps-dev): Bump jest-preset-angular from 14.5.1 to 14.5.3 in /src-ui in the frontend-jest-dependencies group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9255](https://github.com/paperless-ngx/paperless-ngx/pull/9255)) | ||||
| -   Chore(deps): Bump rxjs from 7.8.1 to 7.8.2 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9258](https://github.com/paperless-ngx/paperless-ngx/pull/9258)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.13.0 to 22.13.5 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#9257](https://github.com/paperless-ngx/paperless-ngx/pull/9257)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 22 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9254](https://github.com/paperless-ngx/paperless-ngx/pull/9254)) | ||||
| -   Feature: Switch webserver to granian [@stumpylog](https://github.com/stumpylog) ([#9218](https://github.com/paperless-ngx/paperless-ngx/pull/9218)) | ||||
| -   Enhancement: relocate and smaller upload widget, dont limit upload list [@shamoon](https://github.com/shamoon) ([#9244](https://github.com/paperless-ngx/paperless-ngx/pull/9244)) | ||||
| -   Enhancement: run tasks from system status, report sanity check, simpler classifier check, styling updates [@shamoon](https://github.com/shamoon) ([#9106](https://github.com/paperless-ngx/paperless-ngx/pull/9106)) | ||||
| -   Chore: Switch remote version check to HTTPx [@stumpylog](https://github.com/stumpylog) ([#9232](https://github.com/paperless-ngx/paperless-ngx/pull/9232)) | ||||
| -   Fix: cleanup saved view references on custom field deletion, auto-refresh views, show error on saved view save [@shamoon](https://github.com/shamoon) ([#9225](https://github.com/paperless-ngx/paperless-ngx/pull/9225)) | ||||
| -   Fix: revert thumbnail CSS workaround in favor of GPU workaround [@shamoon](https://github.com/shamoon) ([#9219](https://github.com/paperless-ngx/paperless-ngx/pull/9219)) | ||||
| -   Chore: Reduce imports for a slight memory improvement [@stumpylog](https://github.com/stumpylog) ([#9217](https://github.com/paperless-ngx/paperless-ngx/pull/9217)) | ||||
| -   Enhancement: include celery log in logs view [@shamoon](https://github.com/shamoon) ([#9214](https://github.com/paperless-ngx/paperless-ngx/pull/9214)) | ||||
| -   Enhancement: support default groups for regular and social account signup, syncing on login [@shamoon](https://github.com/shamoon) ([#9039](https://github.com/paperless-ngx/paperless-ngx/pull/9039)) | ||||
| -   Enhancement: allow disabling the filesystem consumer [@shamoon](https://github.com/shamoon) ([#9199](https://github.com/paperless-ngx/paperless-ngx/pull/9199)) | ||||
| -   Fix: correct split confirm removal [@shamoon](https://github.com/shamoon) ([#9195](https://github.com/paperless-ngx/paperless-ngx/pull/9195)) | ||||
| -   Feature: email document [@shamoon](https://github.com/shamoon) ([#8950](https://github.com/paperless-ngx/paperless-ngx/pull/8950)) | ||||
| -   Enhancement: webui workflowtrigger source option [@shamoon](https://github.com/shamoon) ([#9170](https://github.com/paperless-ngx/paperless-ngx/pull/9170)) | ||||
| -   Fix: saved views do not return to default display fields after setting and then removing [@shamoon](https://github.com/shamoon) ([#9168](https://github.com/paperless-ngx/paperless-ngx/pull/9168)) | ||||
| -   Chore(deps): Bump django-filter from 24.3 to 25.1 in the django group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9143](https://github.com/paperless-ngx/paperless-ngx/pull/9143)) | ||||
| -   Chore(deps-dev): Bump mkdocs-material from 9.6.3 to 9.6.4 in the development group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9142](https://github.com/paperless-ngx/paperless-ngx/pull/9142)) | ||||
| -   Fix: correct logged number of deleted documents on trash empty [@shamoon](https://github.com/shamoon) ([#9148](https://github.com/paperless-ngx/paperless-ngx/pull/9148)) | ||||
| -   Fix: include account confirm email allauth URL [@shamoon](https://github.com/shamoon) ([#9147](https://github.com/paperless-ngx/paperless-ngx/pull/9147)) | ||||
| -   Fix: remove additional scrollbar from popup preview [@shamoon](https://github.com/shamoon) ([#9140](https://github.com/paperless-ngx/paperless-ngx/pull/9140)) | ||||
| -   Fix: wrap selected display fields [@shamoon](https://github.com/shamoon) ([#9139](https://github.com/paperless-ngx/paperless-ngx/pull/9139)) | ||||
| -   Enhancement: use charfield for webhook url, custom validation [@shamoon](https://github.com/shamoon) ([#9128](https://github.com/paperless-ngx/paperless-ngx/pull/9128)) | ||||
| -   Fix: reset documents sort field if user deletes the custom field [@shamoon](https://github.com/shamoon) ([#9127](https://github.com/paperless-ngx/paperless-ngx/pull/9127)) | ||||
| -   Chore: more efficient select cf update handler [@shamoon](https://github.com/shamoon) ([#9099](https://github.com/paperless-ngx/paperless-ngx/pull/9099)) | ||||
| -   Fix: limit document title length in workflows [@shamoon](https://github.com/shamoon) ([#9085](https://github.com/paperless-ngx/paperless-ngx/pull/9085)) | ||||
| -   Feature: Chinese Traditional translation [@LokiHung](https://github.com/LokiHung) ([#9076](https://github.com/paperless-ngx/paperless-ngx/pull/9076)) | ||||
| -   Enhancement: Use cached sessions for a minor performance improvement [@stumpylog](https://github.com/stumpylog) ([#9074](https://github.com/paperless-ngx/paperless-ngx/pull/9074)) | ||||
| -   Chore(deps): Bump the small-changes group with 7 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9064](https://github.com/paperless-ngx/paperless-ngx/pull/9064)) | ||||
| -   Chore(deps-dev): Bump the development group with 3 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9061](https://github.com/paperless-ngx/paperless-ngx/pull/9061)) | ||||
| -   Chore(deps): Bump the django group across 1 directory with 2 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9065](https://github.com/paperless-ngx/paperless-ngx/pull/9065)) | ||||
| -   Chore(deps): Bump drf-spectacular-sidecar from 2024.11.1 to 2025.2.1 in the major-versions group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9063](https://github.com/paperless-ngx/paperless-ngx/pull/9063)) | ||||
| -   Feature: openapi spec, full api browser [@shamoon](https://github.com/shamoon) ([#8948](https://github.com/paperless-ngx/paperless-ngx/pull/8948)) | ||||
| -   Fix: include doc link input import in custom fields query dropdown [@shamoon](https://github.com/shamoon) ([#9058](https://github.com/paperless-ngx/paperless-ngx/pull/9058)) | ||||
| -   Enhancement: filter by file type [@shamoon](https://github.com/shamoon) ([#8946](https://github.com/paperless-ngx/paperless-ngx/pull/8946)) | ||||
| -   Enhancement: add layout options for email conversion [@RazielleS](https://github.com/RazielleS) ([#8907](https://github.com/paperless-ngx/paperless-ngx/pull/8907)) | ||||
| -   Chore: Enable ruff FBT [@gothicVI](https://github.com/gothicVI) ([#8645](https://github.com/paperless-ngx/paperless-ngx/pull/8645)) | ||||
| -   Feature: better toast notifications management [@shamoon](https://github.com/shamoon) ([#8980](https://github.com/paperless-ngx/paperless-ngx/pull/8980)) | ||||
| -   Enhancement: date picker and date filter dropdown improvements [@shamoon](https://github.com/shamoon) ([#9033](https://github.com/paperless-ngx/paperless-ngx/pull/9033)) | ||||
| -   Fix: deselect and trigger refresh for deleted documents from bulk operations with delete originals [@shamoon](https://github.com/shamoon) ([#8996](https://github.com/paperless-ngx/paperless-ngx/pull/8996)) | ||||
| -   Tweak: improve date matching regex for dates after numbers [@XstreamGit](https://github.com/XstreamGit) ([#8964](https://github.com/paperless-ngx/paperless-ngx/pull/8964)) | ||||
| -   Tweak: more accurate classifier last trained time [@shamoon](https://github.com/shamoon) ([#9004](https://github.com/paperless-ngx/paperless-ngx/pull/9004)) | ||||
| -   Enhancement: allow setting default pdf zoom [@shamoon](https://github.com/shamoon) ([#9017](https://github.com/paperless-ngx/paperless-ngx/pull/9017)) | ||||
| -   Chore(deps-dev): Bump the development group with 2 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#9013](https://github.com/paperless-ngx/paperless-ngx/pull/9013)) | ||||
| -   Chore(deps): Bump django-soft-delete from 1.0.16 to 1.0.18 in the django group @[dependabot[bot]](https://github.com/apps/dependabot) ([#9014](https://github.com/paperless-ngx/paperless-ngx/pull/9014)) | ||||
| -   Fix: allow empty email in profile [@shamoon](https://github.com/shamoon) ([#9012](https://github.com/paperless-ngx/paperless-ngx/pull/9012)) | ||||
| -   Chore(deps): Bump uuid from 11.0.2 to 11.0.5 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#8992](https://github.com/paperless-ngx/paperless-ngx/pull/8992)) | ||||
| -   Chore(deps-dev): Bump @codecov/webpack-plugin from 1.2.1 to 1.8.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#8991](https://github.com/paperless-ngx/paperless-ngx/pull/8991)) | ||||
| -   Chore(deps-dev): Bump @playwright/test from 1.48.2 to 1.50.1 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#8993](https://github.com/paperless-ngx/paperless-ngx/pull/8993)) | ||||
| -   Chore(deps-dev): Bump @types/node from 22.8.6 to 22.13.0 in /src-ui @[dependabot[bot]](https://github.com/apps/dependabot) ([#8989](https://github.com/paperless-ngx/paperless-ngx/pull/8989)) | ||||
| -   Chore(deps-dev): Bump the frontend-eslint-dependencies group in /src-ui with 4 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#8988](https://github.com/paperless-ngx/paperless-ngx/pull/8988)) | ||||
| -   Chore(deps): Bump the frontend-angular-dependencies group in /src-ui with 23 updates @[dependabot[bot]](https://github.com/apps/dependabot) ([#8986](https://github.com/paperless-ngx/paperless-ngx/pull/8986)) | ||||
| </details> | ||||
|  | ||||
| ## paperless-ngx 2.14.7 | ||||
|  | ||||
| ### Features | ||||
| @@ -4865,9 +5699,6 @@ This release contains new database migrations. | ||||
|     Paperless will continue to work with WSGI, but you will not get any | ||||
|     status notifications. | ||||
|  | ||||
|     Apache `mod_wsgi` users, see | ||||
|     [this note](faq.md#how-do-i-get-websocket-support-with-apache-mod_wsgi). | ||||
|  | ||||
| -   Paperless now offers suggestions for tags, correspondents and types | ||||
|     on the document detail page. | ||||
|  | ||||
| @@ -5449,7 +6280,6 @@ primarily. | ||||
|         a very good job at ocr'ing a document with the default | ||||
|         language. Certain language specifics such as umlauts may not get | ||||
|         picked up properly. | ||||
|     -   `PAPERLESS_DEBUG` defaults to `false`. | ||||
|     -   The presence of `PAPERLESS_DBHOST` now determines whether to use | ||||
|         PostgreSQL or SQLite. | ||||
|     -   `PAPERLESS_OCR_THREADS` is gone and replaced with | ||||
| @@ -5670,11 +6500,12 @@ primarily. | ||||
|         who are doing active development on Paperless using the Docker | ||||
|         environment: | ||||
|         [#376](https://github.com/the-paperless-project/paperless/pull/376). | ||||
| -   You now also have the ability to customise the interface to your | ||||
| -   ~~You now also have the ability to customise the interface to your | ||||
|     heart's content by creating a file called `overrides.css` and/or | ||||
|     `overrides.js` in the root of your media directory. Thanks to [Mark | ||||
|     McFate](https://github.com/SummittDweller) for this idea: | ||||
|     [#371](https://github.com/the-paperless-project/paperless/issues/371) | ||||
|     [#371](https://github.com/the-paperless-project/paperless/issues/371)~~ | ||||
|     (Not supported by Paperless-ngx) | ||||
|  | ||||
| ### 2.0.0 | ||||
|  | ||||
|   | ||||
| @@ -50,47 +50,48 @@ matcher. | ||||
|  | ||||
| ### Database | ||||
|  | ||||
| By default, Paperless uses **SQLite** with a database stored at `data/db.sqlite3`. | ||||
| To switch to **PostgreSQL** or **MariaDB**, set [`PAPERLESS_DBHOST`](#PAPERLESS_DBHOST) and optionally configure other | ||||
| database-related environment variables. | ||||
|  | ||||
| #### [`PAPERLESS_DBHOST=<hostname>`](#PAPERLESS_DBHOST) {#PAPERLESS_DBHOST} | ||||
|  | ||||
| : If unset, Paperless uses **SQLite** by default. | ||||
|  | ||||
|     Set `PAPERLESS_DBHOST` to switch to PostgreSQL or MariaDB instead. | ||||
|  | ||||
| #### [`PAPERLESS_DBENGINE=<engine_name>`](#PAPERLESS_DBENGINE) {#PAPERLESS_DBENGINE} | ||||
|  | ||||
| : Optional, gives the ability to choose Postgres or MariaDB for | ||||
| database engine. Available options are `postgresql` and | ||||
| `mariadb`. | ||||
| : Optional. Specifies the database engine to use when connecting to a remote database. | ||||
| Available options are `postgresql` and `mariadb`. | ||||
|  | ||||
|     Default is `postgresql`. | ||||
|     Defaults to `postgresql` if `PAPERLESS_DBHOST` is set. | ||||
|  | ||||
|     !!! warning | ||||
|  | ||||
|         Using MariaDB comes with some caveats. See [MySQL Caveats](advanced_usage.md#mysql-caveats). | ||||
|  | ||||
| #### [`PAPERLESS_DBHOST=<hostname>`](#PAPERLESS_DBHOST) {#PAPERLESS_DBHOST} | ||||
|  | ||||
| : By default, sqlite is used as the database backend. This can be | ||||
| changed here. | ||||
|  | ||||
|     Set PAPERLESS_DBHOST and another database will be used instead of | ||||
|     sqlite. | ||||
|  | ||||
| #### [`PAPERLESS_DBPORT=<port>`](#PAPERLESS_DBPORT) {#PAPERLESS_DBPORT} | ||||
|  | ||||
| : Adjust port if necessary. | ||||
| : Port to use when connecting to PostgreSQL or MariaDB. | ||||
|  | ||||
|     Default is 5432. | ||||
|     Default is `5432` for PostgreSQL and `3306` for MariaDB. | ||||
|  | ||||
| #### [`PAPERLESS_DBNAME=<name>`](#PAPERLESS_DBNAME) {#PAPERLESS_DBNAME} | ||||
|  | ||||
| : Database name in PostgreSQL or MariaDB. | ||||
| : Name of the database to connect to when using PostgreSQL or MariaDB. | ||||
|  | ||||
|     Defaults to "paperless". | ||||
|  | ||||
| #### [`PAPERLESS_DBUSER=<name>`](#PAPERLESS_DBUSER) {#PAPERLESS_DBUSER} | ||||
|  | ||||
| : Database user in PostgreSQL or MariaDB. | ||||
| : Username for authenticating with the PostgreSQL or MariaDB database. | ||||
|  | ||||
|     Defaults to "paperless". | ||||
|  | ||||
| #### [`PAPERLESS_DBPASS=<password>`](#PAPERLESS_DBPASS) {#PAPERLESS_DBPASS} | ||||
|  | ||||
| : Database password for PostgreSQL or MariaDB. | ||||
| : Password for the PostgreSQL or MariaDB database user. | ||||
|  | ||||
|     Defaults to "paperless". | ||||
|  | ||||
| @@ -110,20 +111,20 @@ changed here. | ||||
|  | ||||
| #### [`PAPERLESS_DBSSLROOTCERT=<ca-path>`](#PAPERLESS_DBSSLROOTCERT) {#PAPERLESS_DBSSLROOTCERT} | ||||
|  | ||||
| : SSL root certificate path | ||||
| : Path to the SSL root certificate used to verify the database server. | ||||
|  | ||||
|     See [the official documentation about | ||||
|     sslmode for PostgreSQL](https://www.postgresql.org/docs/current/libpq-ssl.html). | ||||
|     Changes path of `root.crt`. | ||||
|     Changes the location of `root.crt`. | ||||
|  | ||||
|     See [the official documentation about | ||||
|     sslmode for MySQL and MariaDB](https://dev.mysql.com/doc/refman/8.0/en/connection-options.html#option_general_ssl-ca). | ||||
|  | ||||
|     Defaults to unset, using the documented path in the home directory. | ||||
|     Defaults to unset, using the standard location in the home directory. | ||||
|  | ||||
| #### [`PAPERLESS_DBSSLCERT=<client-cert-path>`](#PAPERLESS_DBSSLCERT) {#PAPERLESS_DBSSLCERT} | ||||
|  | ||||
| : SSL client certificate path | ||||
| : Path to the client SSL certificate used when connecting securely. | ||||
|  | ||||
|     See [the official documentation about | ||||
|     sslmode for PostgreSQL](https://www.postgresql.org/docs/current/libpq-ssl.html). | ||||
| @@ -131,13 +132,13 @@ changed here. | ||||
|     See [the official documentation about | ||||
|     sslmode for MySQL and MariaDB](https://dev.mysql.com/doc/refman/8.0/en/connection-options.html#option_general_ssl-cert). | ||||
|  | ||||
|     Changes path of `postgresql.crt`. | ||||
|     Changes the location of `postgresql.crt`. | ||||
|  | ||||
|     Defaults to unset, using the documented path in the home directory. | ||||
|     Defaults to unset, using the standard location in the home directory. | ||||
|  | ||||
| #### [`PAPERLESS_DBSSLKEY=<client-cert-key>`](#PAPERLESS_DBSSLKEY) {#PAPERLESS_DBSSLKEY} | ||||
|  | ||||
| : SSL client key path | ||||
| : Path to the client SSL private key used when connecting securely. | ||||
|  | ||||
|     See [the official documentation about | ||||
|     sslmode for PostgreSQL](https://www.postgresql.org/docs/current/libpq-ssl.html). | ||||
| @@ -145,17 +146,70 @@ changed here. | ||||
|     See [the official documentation about | ||||
|     sslmode for MySQL and MariaDB](https://dev.mysql.com/doc/refman/8.0/en/connection-options.html#option_general_ssl-key). | ||||
|  | ||||
|     Changes path of `postgresql.key`. | ||||
|     Changes the location of `postgresql.key`. | ||||
|  | ||||
|     Defaults to unset, using the documented path in the home directory. | ||||
|     Defaults to unset, using the standard location in the home directory. | ||||
|  | ||||
| #### [`PAPERLESS_DB_TIMEOUT=<int>`](#PAPERLESS_DB_TIMEOUT) {#PAPERLESS_DB_TIMEOUT} | ||||
|  | ||||
| : Amount of time for a database connection to wait for the database to | ||||
| unlock. Mostly applicable for sqlite based installation. Consider changing | ||||
| to postgresql if you are having concurrency problems with sqlite. | ||||
| : Sets how long a database connection should wait before timing out. | ||||
|  | ||||
|     Defaults to unset, keeping the Django defaults. | ||||
|     For SQLite, this sets how long to wait if the database is locked. | ||||
|     For PostgreSQL or MariaDB, this sets the connection timeout. | ||||
|  | ||||
|     Defaults to unset, which uses Django’s built-in defaults. | ||||
|  | ||||
| #### [`PAPERLESS_DB_POOLSIZE=<int>`](#PAPERLESS_DB_POOLSIZE) {#PAPERLESS_DB_POOLSIZE} | ||||
|  | ||||
| : Defines the maximum number of database connections to keep in the pool. | ||||
|  | ||||
|     Only applies to PostgreSQL. This setting is ignored for other database engines. | ||||
|  | ||||
|     The value must be greater than or equal to 1 to be used. | ||||
|     Defaults to unset, which disables connection pooling. | ||||
|  | ||||
|     !!! note | ||||
|  | ||||
|     A small pool is typically sufficient — for example, a size of 4. | ||||
|     Make sure your PostgreSQL server's max_connections setting is large enough to handle: | ||||
|     ```(Paperless workers + Celery workers) × pool size + safety margin``` | ||||
|     For example, with 4 Paperless workers and 2 Celery workers, and a pool size of 4: | ||||
|     (4 + 2) × 4 + 10 = 34 connections required. | ||||
|  | ||||
| #### [`PAPERLESS_DB_READ_CACHE_ENABLED=<bool>`](#PAPERLESS_DB_READ_CACHE_ENABLED) {#PAPERLESS_DB_READ_CACHE_ENABLED} | ||||
|  | ||||
| : Caches the database read query results into Redis. This can significantly improve application response times by caching database queries, at the cost of slightly increased memory usage. | ||||
|  | ||||
|     Defaults to `false`. | ||||
|  | ||||
|     !!! danger | ||||
|  | ||||
|     **Do not modify the database outside the application while it is running.** | ||||
|     This includes actions such as restoring a backup, upgrading the database, or performing manual inserts. All external modifications must be done **only when the application is stopped**. | ||||
|     After making any such changes, you **must invalidate the DB read cache** using the `invalidate_cachalot` management command. | ||||
|  | ||||
| #### [`PAPERLESS_READ_CACHE_TTL=<int>`](#PAPERLESS_READ_CACHE_TTL) {#PAPERLESS_READ_CACHE_TTL} | ||||
|  | ||||
| : Specifies how long (in seconds) read data should be cached. | ||||
|  | ||||
|     Allowed values are between `1` (one second) and `31536000` (one year). Defaults to `3600` (one hour). | ||||
|  | ||||
|     !!! warning | ||||
|  | ||||
|     A high TTL increases memory usage over time. Memory may be used until end of TTL, even if the cache is invalidated with the `invalidate_cachalot` command. | ||||
|  | ||||
| In case of an out-of-memory (OOM) situation, Redis may stop accepting new data — including cache entries, scheduled tasks, and documents to consume. | ||||
| If your system has limited RAM, consider configuring a dedicated Redis instance for the read cache, with a memory limit and the eviction policy set to `allkeys-lru`. | ||||
| For more details, refer to the [Redis eviction policy documentation](https://redis.io/docs/latest/develop/reference/eviction/), and see the `PAPERLESS_READ_CACHE_REDIS_URL` setting to specify a separate Redis broker. | ||||
|  | ||||
| #### [`PAPERLESS_READ_CACHE_REDIS_URL=<url>`](#PAPERLESS_READ_CACHE_REDIS_URL) {#PAPERLESS_READ_CACHE_REDIS_URL} | ||||
|  | ||||
| : Defines the Redis instance used for the read cache. | ||||
|  | ||||
|     Defaults to `None`. | ||||
|  | ||||
|     !!! Note | ||||
|     If this value is not set, the same Redis instance used for scheduled tasks will be used for caching as well. | ||||
|  | ||||
| ## Optional Services | ||||
|  | ||||
| @@ -200,7 +254,7 @@ and watch out for indentation if editing the YAML file. | ||||
|  | ||||
| ### Email Parsing | ||||
|  | ||||
| #### [`PAPERLESS_EMAIL_PARSE_DEFAULT_LAYOUT=<int>`(#PAPERLESS_EMAIL_PARSE_DEFAULT_LAYOUT) {#PAPERLESS_EMAIL_PARSE_DEFAULT_LAYOUT} | ||||
| #### [`PAPERLESS_EMAIL_PARSE_DEFAULT_LAYOUT=<int>`](#PAPERLESS_EMAIL_PARSE_DEFAULT_LAYOUT) {#PAPERLESS_EMAIL_PARSE_DEFAULT_LAYOUT} | ||||
|  | ||||
| : The default layout to use for emails that are consumed as documents. Must be one of the integer choices below. Note that mail | ||||
| rules can specify this setting, thus this fallback is used for the default selection and for .eml files consumed by other means. | ||||
| @@ -404,7 +458,7 @@ set this value to /paperless. No trailing slash! | ||||
| #### [`PAPERLESS_STATIC_URL=<path>`](#PAPERLESS_STATIC_URL) {#PAPERLESS_STATIC_URL} | ||||
|  | ||||
| : Override the STATIC_URL here. Unless you're hosting Paperless off a | ||||
| subdomain like /paperless/, you probably don't need to change this. | ||||
| specific path like /paperless/, you probably don't need to change this. | ||||
| If you do change it, be sure to include the trailing slash. | ||||
|  | ||||
|     Defaults to "/static/". | ||||
| @@ -629,7 +683,13 @@ If both the [PAPERLESS_ACCOUNT_DEFAULT_GROUPS](#PAPERLESS_ACCOUNT_DEFAULT_GROUPS | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     If you do not have a working email server set up you should set this to 'none'. | ||||
|     If you do not have a working email server set up this will be set to 'none'. | ||||
|  | ||||
| #### [`PAPERLESS_ACCOUNT_EMAIL_UNKNOWN_ACCOUNTS=<bool>`](#PAPERLESS_ACCOUNT_EMAIL_UNKNOWN_ACCOUNTS) {#PAPERLESS_ACCOUNT_EMAIL_UNKNOWN_ACCOUNTS} | ||||
|  | ||||
| : See the relevant [django-allauth documentation](https://docs.allauth.org/en/latest/account/configuration.html) | ||||
|  | ||||
|     Defaults to True (from allauth) | ||||
|  | ||||
| #### [`PAPERLESS_DISABLE_REGULAR_LOGIN=<bool>`](#PAPERLESS_DISABLE_REGULAR_LOGIN) {#PAPERLESS_DISABLE_REGULAR_LOGIN} | ||||
|  | ||||
| @@ -960,6 +1020,22 @@ still perform some basic text pre-processing before matching. | ||||
|  | ||||
|     Defaults to 1. | ||||
|  | ||||
| #### [`PAPERLESS_DATE_PARSER_LANGUAGES=<lang>`](#PAPERLESS_DATE_PARSER_LANGUAGES) {#PAPERLESS_DATE_PARSER_LANGUAGES} | ||||
|  | ||||
| Specifies which language Paperless should use when parsing dates from documents. | ||||
|  | ||||
|     This should be a language code supported by the dateparser library, | ||||
|     for example: "en", or a combination such as "en+de". | ||||
|     Locales are also supported (e.g., "en-AU"). | ||||
|     Multiple languages can be combined using "+", for example: "en+de" or "en-AU+de". | ||||
|     For valid values, refer to the list of supported languages and locales in the [dateparser documentation](https://dateparser.readthedocs.io/en/latest/supported_locales.html). | ||||
|  | ||||
|     Set this to match the languages in which most of your documents are written. | ||||
|     If not set, Paperless will attempt to infer the language(s) from the OCR configuration (`PAPERLESS_OCR_LANGUAGE`). | ||||
|  | ||||
| !!! note | ||||
| This format differs from the `PAPERLESS_OCR_LANGUAGE` setting, which uses ISO 639-2 codes (3 letters, e.g., "eng+deu" for Tesseract OCR). | ||||
|  | ||||
| #### [`PAPERLESS_EMAIL_TASK_CRON=<cron expression>`](#PAPERLESS_EMAIL_TASK_CRON) {#PAPERLESS_EMAIL_TASK_CRON} | ||||
|  | ||||
| : Configures the scheduled email fetching frequency. The value | ||||
| @@ -1057,9 +1133,9 @@ be used with caution! | ||||
|  | ||||
| ## Document Consumption {#consume_config} | ||||
|  | ||||
| #### [`PAPERLESS_CONSUMER_DISABLE=<bool>`](#PAPERLESS_CONSUMER_DISABLE) {#PAPERLESS_CONSUMER_DISABLE} | ||||
| #### [`PAPERLESS_CONSUMER_DISABLE`](#PAPERLESS_CONSUMER_DISABLE) {#PAPERLESS_CONSUMER_DISABLE} | ||||
|  | ||||
| : Completely disable the directory-based consumer in docker. If you don't plan to consume documents | ||||
| : If set (to anything), this completely disables the directory-based consumer in docker. If you don't plan to consume documents | ||||
| via the consumption directory, you can disable the consumer to save resources. | ||||
|  | ||||
| #### [`PAPERLESS_CONSUMER_DELETE_DUPLICATES=<bool>`](#PAPERLESS_CONSUMER_DELETE_DUPLICATES) {#PAPERLESS_CONSUMER_DELETE_DUPLICATES} | ||||
| @@ -1206,6 +1282,30 @@ within your documents. | ||||
|  | ||||
|     Defaults to false. | ||||
|  | ||||
| ## Workflow webhooks | ||||
|  | ||||
| #### [`PAPERLESS_WEBHOOKS_ALLOWED_SCHEMES=<str>`](#PAPERLESS_WEBHOOKS_ALLOWED_SCHEMES) {#PAPERLESS_WEBHOOKS_ALLOWED_SCHEMES} | ||||
|  | ||||
| : A comma-separated list of allowed schemes for webhooks. This setting | ||||
| controls which URL schemes are permitted for webhook URLs. | ||||
|  | ||||
|     Defaults to `http,https`. | ||||
|  | ||||
| #### [`PAPERLESS_WEBHOOKS_ALLOWED_PORTS=<str>`](#PAPERLESS_WEBHOOKS_ALLOWED_PORTS) {#PAPERLESS_WEBHOOKS_ALLOWED_PORTS} | ||||
|  | ||||
| : A comma-separated list of allowed ports for webhooks. This setting | ||||
| controls which ports are permitted for webhook URLs. For example, if you | ||||
| set this to `80,443`, webhooks will only be sent to URLs that use these | ||||
| ports. | ||||
|  | ||||
|     Defaults to empty list, which allows all ports. | ||||
|  | ||||
| #### [`PAPERLESS_WEBHOOKS_ALLOW_INTERNAL_REQUESTS=<bool>`](#PAPERLESS_WEBHOOKS_ALLOW_INTERNAL_REQUESTS) {#PAPERLESS_WEBHOOKS_ALLOW_INTERNAL_REQUESTS} | ||||
|  | ||||
| : If set to false, webhooks cannot be sent to internal URLs (e.g., localhost). | ||||
|  | ||||
|     Defaults to true, which allows internal requests. | ||||
|  | ||||
| ### Polling {#polling} | ||||
|  | ||||
| #### [`PAPERLESS_CONSUMER_POLLING=<num>`](#PAPERLESS_CONSUMER_POLLING) {#PAPERLESS_CONSUMER_POLLING} | ||||
| @@ -1659,6 +1759,11 @@ started by the container. | ||||
|  | ||||
| : Path to an image file in the /media/logo directory, must include 'logo', e.g. `/logo/Atari_logo.svg` | ||||
|  | ||||
| !!! note | ||||
|  | ||||
|     The logo file will be viewable by anyone with access to the Paperless instance login page, | ||||
|     so consider your choice of logo carefully and removing exif data from images before uploading. | ||||
|  | ||||
| #### [`PAPERLESS_ENABLE_UPDATE_CHECK=<bool>`](#PAPERLESS_ENABLE_UPDATE_CHECK) {#PAPERLESS_ENABLE_UPDATE_CHECK} | ||||
|  | ||||
| !!! note | ||||
| @@ -1670,7 +1775,7 @@ started by the container. | ||||
|  | ||||
| ## Email sending | ||||
|  | ||||
| Setting an SMTP server for the backend will allow you to reset your | ||||
| Setting an SMTP server for the backend will allow you to use the Email workflow action, send documents from the UI as well as reset your | ||||
| password. All of these options come from their similarly-named [Django settings](https://docs.djangoproject.com/en/4.2/ref/settings/#email-host) | ||||
|  | ||||
| #### [`PAPERLESS_EMAIL_HOST=<str>`](#PAPERLESS_EMAIL_HOST) {#PAPERLESS_EMAIL_HOST} | ||||
|   | ||||
| @@ -84,7 +84,7 @@ first-time setup. | ||||
|     $ uv run pre-commit install | ||||
|     ``` | ||||
|  | ||||
| 6.  Apply migrations and create a superuser for your development instance: | ||||
| 6.  Apply migrations and create a superuser (also can be done via the web UI) for your development instance: | ||||
|  | ||||
|     ```bash | ||||
|     # src/ | ||||
| @@ -95,13 +95,13 @@ first-time setup. | ||||
|  | ||||
| 7.  You can now either ... | ||||
|  | ||||
|     -   install redis or | ||||
|     -   install Redis or | ||||
|  | ||||
|     -   use the included `scripts/start_services.sh` to use docker to fire | ||||
|         up a redis instance (and some other services such as tika, | ||||
|         gotenberg and a database server) or | ||||
|     -   use the included `scripts/start_services.sh` to use Docker to fire | ||||
|         up a Redis instance (and some other services such as Tika, | ||||
|         Gotenberg and a database server) or | ||||
|  | ||||
|     -   spin up a bare redis container | ||||
|     -   spin up a bare Redis container | ||||
|  | ||||
|         ``` | ||||
|         docker run -d -p 6379:6379 --restart unless-stopped redis:latest | ||||
| @@ -147,7 +147,7 @@ $ ng build --configuration production | ||||
| ### Testing | ||||
|  | ||||
| -   Run `pytest` in the `src/` directory to execute all tests. This also | ||||
|     generates a HTML coverage report. When runnings test, `paperless.conf` | ||||
|     generates a HTML coverage report. When running tests, `paperless.conf` | ||||
|     is loaded as well. However, the tests rely on the default | ||||
|     configuration. This is not ideal. But for now, make sure no settings | ||||
|     except for DEBUG are overridden when testing. | ||||
| @@ -470,9 +470,14 @@ To get started: | ||||
|  | ||||
| 2. VS Code will prompt you with "Reopen in container". Do so and wait for the environment to start. | ||||
|  | ||||
| 3. Initialize the project by running the task **Project Setup: Run all Init Tasks**. This | ||||
| 3. In case your host operating system is Windows: | ||||
|  | ||||
|     - The Source Control view in Visual Studio Code might show: "The detected Git repository is potentially unsafe as the folder is owned by someone other than the current user." Use "Manage Unsafe Repositories" to fix this. | ||||
|     - Git might have detecteded modifications for all files, because Windows is using CRLF line endings. Run `git checkout .` in the containers terminal to fix this issue. | ||||
|  | ||||
| 4. Initialize the project by running the task **Project Setup: Run all Init Tasks**. This | ||||
|    will initialize the database tables and create a superuser. Then you can compile the front end | ||||
|    for production or run the frontend in debug mode. | ||||
|  | ||||
| 4. The project is ready for debugging, start either run the fullstack debug or individual debug | ||||
| 5. The project is ready for debugging, start either run the fullstack debug or individual debug | ||||
|    processes. Yo spin up the project without debugging run the task **Project Start: Run all Services** | ||||
|   | ||||
							
								
								
									
										24
									
								
								docs/faq.md
									
									
									
									
									
								
							
							
						
						
									
										24
									
								
								docs/faq.md
									
									
									
									
									
								
							| @@ -112,30 +112,6 @@ 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. | ||||
|  | ||||
| ## _How do I proxy this with NGINX?_ | ||||
|  | ||||
| **A:** See [the wiki](https://github.com/paperless-ngx/paperless-ngx/wiki/Using-a-Reverse-Proxy-with-Paperless-ngx#nginx). | ||||
|  | ||||
| ## _How do I get WebSocket support with Apache mod_wsgi_? | ||||
|  | ||||
| **A:** `mod_wsgi` by itself does not support ASGI. Paperless will | ||||
| continue to work with WSGI, but certain features such as status | ||||
| notifications about document consumption won't be available. | ||||
|  | ||||
| If you want to continue using `mod_wsgi`, you will have to run an | ||||
| ASGI-enabled web server as well that processes WebSocket connections, | ||||
| and configure Apache to redirect WebSocket connections to this server. | ||||
| Multiple options for ASGI servers exist: | ||||
|  | ||||
| -   `gunicorn` with `uvicorn` as the worker implementation (the default | ||||
|     of paperless) | ||||
| -   `daphne` as a standalone server, which is the reference | ||||
|     implementation for ASGI. | ||||
| -   `uvicorn` as a standalone server | ||||
|  | ||||
| You may also find the [Django documentation](https://docs.djangoproject.com/en/5.1/howto/deployment/asgi/) on ASGI | ||||
| useful to review. | ||||
|  | ||||
| ## _What about the Redis licensing change and using one of the open source forks_? | ||||
|  | ||||
| Currently (October 2024), forks of Redis such as Valkey or Redirect are not officially supported by our upstream | ||||
|   | ||||
| @@ -30,7 +30,7 @@ physical documents into a searchable online archive so you can keep, well, _less | ||||
| -   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. | ||||
| -   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. | ||||
| @@ -197,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 the [Paperless-ngx project at Crowdin](https://crwd.in/paperless-ngx), and thank you! | ||||
| Paperless-ngx is available in many languages that are coordinated on [Crowdin](https://crowdin.com/project/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://crowdin.com/project/paperless-ngx), and thank you! | ||||
|  | ||||
| ## Scanners & Software | ||||
|  | ||||
|   | ||||
| @@ -131,26 +131,11 @@ account. The script essentially automatically performs the steps described in [D | ||||
|     by default but you can change the image to pull from Docker Hub by changing the `image` | ||||
|     line to `image: paperlessngx/paperless-ngx:latest`. | ||||
|  | ||||
| 6.  To be able to login, you will need a "superuser". To create it, | ||||
|     execute the following command: | ||||
| 6.  Run `docker compose up -d`. This will create and start the necessary containers. | ||||
|  | ||||
|     ```shell-session | ||||
|     docker compose run --rm webserver createsuperuser | ||||
|     ``` | ||||
|  | ||||
|     or using docker exec from within the container: | ||||
|  | ||||
|     ```shell-session | ||||
|     python3 manage.py createsuperuser | ||||
|     ``` | ||||
|  | ||||
|     This will guide you through the superuser setup. | ||||
|  | ||||
| 7.  Run `docker compose up -d`. This will create and start the necessary containers. | ||||
|  | ||||
| 8.  Congratulations! Your Paperless-ngx instance should now be accessible at `http://127.0.0.1:8000` | ||||
|     (or similar, depending on your configuration). Use the superuser credentials you have | ||||
|     created in the previous step to login. | ||||
| 7.  Congratulations! Your Paperless-ngx instance should now be accessible at `http://127.0.0.1:8000` | ||||
|     (or similar, depending on your configuration). When you first access the web interface, you will be | ||||
|     prompted to create a superuser account. | ||||
|  | ||||
| ### Build the Docker image yourself {#docker_build} | ||||
|  | ||||
| @@ -386,16 +371,15 @@ are released, dependency support is confirmed, etc. | ||||
|         dependencies for Postgres or Mariadb.  You can select those extras with `--extra <EXTRA>` | ||||
|         or all with `--all-extras` | ||||
|  | ||||
| 9.  Go to `/opt/paperless/src`, and execute the following commands: | ||||
| 9.  Go to `/opt/paperless/src`, and execute the following command: | ||||
|  | ||||
|     ```bash | ||||
|     # This creates the database schema. | ||||
|     sudo -Hu paperless python3 manage.py migrate | ||||
|  | ||||
|     # This creates your first paperless user | ||||
|     sudo -Hu paperless python3 manage.py createsuperuser | ||||
|     ``` | ||||
|  | ||||
|     When you first access the web interface you will be prompted to create a superuser account. | ||||
|  | ||||
| 10. Optional: Test that paperless is working by executing | ||||
|  | ||||
|     ```bash | ||||
| @@ -461,7 +445,7 @@ are released, dependency support is confirmed, etc. | ||||
| 13. Configure ImageMagick to allow processing of PDF documents. Most | ||||
|     distributions have this disabled by default, since PDF documents can | ||||
|     contain malware. If you don't do this, paperless will fall back to | ||||
|     ghostscript for certain steps such as thumbnail generation. | ||||
|     Ghostscript for certain steps such as thumbnail generation. | ||||
|  | ||||
|     Edit `/etc/ImageMagick-6/policy.xml` and adjust | ||||
|  | ||||
| @@ -708,7 +692,8 @@ Paperless runs on Raspberry Pi. However, some things are rather slow on | ||||
| the Pi and configuring some options in paperless can help improve | ||||
| performance immensely: | ||||
|  | ||||
| -   Stick with SQLite to save some resources. | ||||
| -   Stick with SQLite to save some resources. See [troubleshooting](troubleshooting.md#log-reports-creating-paperlesstask-failed) | ||||
|     if you encounter issues with SQLite locking. | ||||
| -   If you do not need the filesystem-based consumer, consider disabling it | ||||
|     entirely by setting [`PAPERLESS_CONSUMER_DISABLE`](configuration.md#PAPERLESS_CONSUMER_DISABLE) to `true`. | ||||
| -   Consider setting [`PAPERLESS_OCR_PAGES`](configuration.md#PAPERLESS_OCR_PAGES) to 1, so that paperless will | ||||
|   | ||||
| @@ -33,7 +33,7 @@ warns that | ||||
| `OCR for XX failed, but we're going to stick with what we've got since FORGIVING_OCR is enabled`, | ||||
| then you might need to install the [Tesseract language | ||||
| files](https://packages.ubuntu.com/search?keywords=tesseract-ocr) | ||||
| marching your document's languages. | ||||
| matching your document's languages. | ||||
|  | ||||
| As an example, if you are running Paperless-ngx from any Ubuntu or | ||||
| Debian box, and your documents are written in Spanish you may need to | ||||
| @@ -130,7 +130,7 @@ command: | ||||
|     - 'gotenberg' | ||||
|     - '--chromium-disable-javascript=true' | ||||
|     - '--chromium-allow-list=file:///tmp/.*' | ||||
|     - '--api-timeout=60' | ||||
|     - '--api-timeout=60s' | ||||
| ``` | ||||
|  | ||||
| ## Permission denied errors in the consumption directory | ||||
| @@ -292,7 +292,9 @@ 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`](configuration.md#PAPERLESS_DB_TIMEOUT) setting to allow more time for the database to | ||||
| unlock. This may have minor performance implications. | ||||
| unlock. Additionally, you can change your SQLite database to use ["Write-Ahead Logging"](https://sqlite.org/wal.html). | ||||
| These changes may have minor performance implications but can help | ||||
| prevent database locking issues. | ||||
|  | ||||
| ## granian fails to start with "is not a valid port number" | ||||
|  | ||||
| @@ -333,7 +335,7 @@ You may see errors when deleting documents like: | ||||
| Data too long for column 'transaction_id' at row 1 | ||||
| ``` | ||||
|  | ||||
| This error can occur in installations which have upgraded from a version of Paperless-ngx that used Django 4 (Paperless-ngx versions prior to v2.13.0) with a MariaDB/MySQL database. Due to the backawards-incompatible change in Django 5, the column "documents_document.transaction_id" will need to be re-created, which can be done with a one-time run of the following management command: | ||||
| This error can occur in installations which have upgraded from a version of Paperless-ngx that used Django 4 (Paperless-ngx versions prior to v2.13.0) with a MariaDB/MySQL database. Due to the backwards-incompatible change in Django 5, the column "documents_document.transaction_id" will need to be re-created, which can be done with a one-time run of the following management command: | ||||
|  | ||||
| ```shell-session | ||||
| $ python3 manage.py convert_mariadb_uuid | ||||
|   | ||||
							
								
								
									
										209
									
								
								docs/usage.md
									
									
									
									
									
								
							
							
						
						
									
										209
									
								
								docs/usage.md
									
									
									
									
									
								
							| @@ -30,6 +30,9 @@ Each document has data fields that you can assign to them: | ||||
| -   A _document type_ is used to demarcate the type of a document such | ||||
|     as letter, bank statement, invoice, contract, etc. It is used to | ||||
|     identify what a document is about. | ||||
| -   The document _storage path_ is the location where the document files | ||||
|     are stored. See [Storage Paths](advanced_usage.md#storage-paths) for | ||||
|     more information. | ||||
| -   The _date added_ of a document is the date the document was scanned | ||||
|     into paperless. You cannot and should not change this date. | ||||
| -   The _date created_ of a document is the date the document was | ||||
| @@ -89,7 +92,17 @@ and more. These areas allow you to view, add, edit, delete and manage permission | ||||
| for these objects. You can also manage saved views, mail accounts, mail rules, | ||||
| workflows and more from the management sections. | ||||
|  | ||||
| ## Adding documents to paperless | ||||
| ### Nested Tags | ||||
|  | ||||
| Paperless-ngx v2.19 introduces support for nested tags, allowing you to create a | ||||
| hierarchy of tags, which may be useful for organizing your documents. Tags can | ||||
| have a 'parent' tag, creating a tree-like structure, to a maximum depth of 5. When | ||||
| a tag is added to a document, all of its parent tags are also added automatically | ||||
| and similarly, when a tag is removed from a document, all of its child tags are | ||||
| also removed. Additionally, assigning a parent to an existing tag will automatically | ||||
| update all documents that have this tag assigned, adding the parent tag as well. | ||||
|  | ||||
| ## Adding documents to Paperless-ngx | ||||
|  | ||||
| Once you've got Paperless setup, you need to start feeding documents | ||||
| into it. When adding documents to paperless, it will perform the | ||||
| @@ -115,7 +128,8 @@ following operations on your documents: | ||||
|  | ||||
|     No matter which options you choose, Paperless will always store the | ||||
|     original document that it found in the consumption directory or in the | ||||
|     mail and will never overwrite that document. Archived versions are | ||||
|     mail and will never overwrite that document (except when using certain | ||||
|     document actions, which make that clear). Archived versions are | ||||
|     stored alongside the original versions. Any files found in the | ||||
|     consumption directory will stored inside the Paperless-ngx file | ||||
|     structure and will not be retained in the consumption directory. | ||||
| @@ -159,7 +173,7 @@ process. | ||||
| Please see [the wiki](https://github.com/paperless-ngx/paperless-ngx/wiki/Related-Projects) for a user-maintained list of related projects and | ||||
| software (e.g. for mobile devices) that is compatible with Paperless-ngx. | ||||
|  | ||||
| ### Email {#usage-email} | ||||
| ### Incoming Email {#incoming-mail} | ||||
|  | ||||
| You can tell paperless-ngx to consume documents from your email | ||||
| accounts. This is a very flexible and powerful feature, if you regularly | ||||
| @@ -247,6 +261,10 @@ different means. These are as follows: | ||||
| Paperless is set up to check your mails every 10 minutes. This can be | ||||
| configured via [`PAPERLESS_EMAIL_TASK_CRON`](configuration.md#PAPERLESS_EMAIL_TASK_CRON) | ||||
|  | ||||
| #### Processed Mail | ||||
|  | ||||
| Paperless keeps track of emails it has processed in order to avoid processing the same mail multiple times. This uses the message `UID` provided by the mail server, which should be unique for each message. You can view and manage processed mails from the web UI under Mail > Processed Mails. If you need to re-process a message, you can delete the corresponding processed mail entry, which will allow Paperless-ngx to process the email again the next time the mail fetch task runs. | ||||
|  | ||||
| #### OAuth Email Setup | ||||
|  | ||||
| Paperless-ngx supports OAuth2 authentication for Gmail and Outlook email accounts. To set up an email account with OAuth2, you will need to create a 'developer' app with the respective provider and obtain the client ID and client secret and set the appropriate [configuration variables](configuration.md#email_oauth). You will also need to set either [`PAPERLESS_OAUTH_CALLBACK_BASE_URL`](configuration.md#PAPERLESS_OAUTH_CALLBACK_BASE_URL) or [`PAPERLESS_URL`](configuration.md#PAPERLESS_URL) to the correct value for the OAuth2 flow to work correctly. | ||||
| @@ -260,6 +278,31 @@ Once setup, navigating to the email settings page in Paperless-ngx will allow yo | ||||
| You can also submit a document using the REST API, see [POSTing documents](api.md#file-uploads) | ||||
| for details. | ||||
|  | ||||
| ## Sharing documents from Paperless-ngx | ||||
|  | ||||
| Paperless-ngx supports sharing documents with other users by assigning them [permissions](#object-permissions) | ||||
| to the document. Document files can also be shared externally via [share links](#share-links), [email](#email-sharing) | ||||
| or using [email](#workflow-action-email) or [webhook](#workflow-action-webhook) actions in workflows. | ||||
|  | ||||
| ### Share Links | ||||
|  | ||||
| "Share links" are shareable public links to files and can be created and managed under the 'Send' button 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. | ||||
|  | ||||
| ### Email Sharing {#email-sharing} | ||||
|  | ||||
| Paperless-ngx supports directly sending documents via email. If an email server has been [configured](configuration.md#email-sending) | ||||
| the "Send" button on the document detail page will include an "Email" option. You can also share files via email automatically by using | ||||
| a [workflow action](#workflow-action-email). | ||||
|  | ||||
| ## Permissions | ||||
|  | ||||
| Permissions in Paperless-ngx are based around ['global' permissions](#global-permissions) as well as | ||||
| @@ -312,25 +355,25 @@ Global permissions define what areas of the app and API endpoints users can acce | ||||
| 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 | Add, edit, delete or view Correspondents.                                                                                                                                | | ||||
| | CustomField   | Add, edit, delete or view Custom Fields.                                                                                                                                 | | ||||
| | Document      | Add, edit, delete or view Documents.                                                                                                                                     | | ||||
| | DocumentType  | Add, edit, delete or view Document Types.                                                                                                                                | | ||||
| | Group         | Add, edit, delete or view Groups.                                                                                                                                        | | ||||
| | MailAccount   | Add, edit, delete or view Mail Accounts.                                                                                                                                 | | ||||
| | MailRule      | Add, edit, delete or view Mail Rules.                                                                                                                                    | | ||||
| | Note          | Add, edit, delete or view Notes.                                                                                                                                         | | ||||
| | PaperlessTask | View or dismiss (_Change_) File Tasks.                                                                                                                                   | | ||||
| | SavedView     | Add, edit, delete or view Saved Views.                                                                                                                                   | | ||||
| | ShareLink     | Add, delete or view Share Links.                                                                                                                                         | | ||||
| | StoragePath   | Add, edit, delete or view Storage Paths.                                                                                                                                 | | ||||
| | Tag           | Add, edit, delete or view Tags.                                                                                                                                          | | ||||
| | UISettings    | Add, edit, delete or view the UI settings that are used by the web app.<br/>:warning: **Users that will access the web UI must be granted at least _View_ permissions.** | | ||||
| | User          | Add, edit, delete or view Users.                                                                                                                                         | | ||||
| | Workflow      | 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.          | | ||||
| | Type          | Details                                                                                                                                                                                                                         | | ||||
| | ------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | | ||||
| | AppConfig     | _Change_ or higher permissions grants access to the "Application Configuration" area.                                                                                                                                           | | ||||
| | Correspondent | Add, edit, delete or view Correspondents.                                                                                                                                                                                       | | ||||
| | CustomField   | Add, edit, delete or view Custom Fields.                                                                                                                                                                                        | | ||||
| | Document      | Add, edit, delete or view Documents.                                                                                                                                                                                            | | ||||
| | DocumentType  | Add, edit, delete or view Document Types.                                                                                                                                                                                       | | ||||
| | Group         | Add, edit, delete or view Groups.                                                                                                                                                                                               | | ||||
| | MailAccount   | Add, edit, delete or view Mail Accounts.                                                                                                                                                                                        | | ||||
| | MailRule      | Add, edit, delete or view Mail Rules.                                                                                                                                                                                           | | ||||
| | Note          | Add, edit, delete or view Notes.                                                                                                                                                                                                | | ||||
| | PaperlessTask | View or dismiss (_Change_) File Tasks.                                                                                                                                                                                          | | ||||
| | SavedView     | Add, edit, delete or view Saved Views.                                                                                                                                                                                          | | ||||
| | ShareLink     | Add, delete or view Share Links.                                                                                                                                                                                                | | ||||
| | StoragePath   | Add, edit, delete or view Storage Paths.                                                                                                                                                                                        | | ||||
| | Tag           | Add, edit, delete or view Tags.                                                                                                                                                                                                 | | ||||
| | UISettings    | Add, edit, delete or view the UI settings that are used by the web app.<br/>:warning: **Users that will access the web UI must be granted at least _View_ permissions.**                                                        | | ||||
| | User          | Add, edit, delete or view Users.                                                                                                                                                                                                | | ||||
| | Workflow      | Add, edit, delete or view Workflows.<br/>Note that Workflows are global; all users who can access workflows see the same set. Workflows have other permission implications — see [Workflow permissions](#workflow-permissions). | | ||||
|  | ||||
| #### Detailed Explanation of Object Permissions {#object-permissions} | ||||
|  | ||||
| @@ -369,7 +412,7 @@ fields and permissions, which will be merged. | ||||
|  | ||||
| ### Workflow Triggers | ||||
|  | ||||
| #### Types | ||||
| #### Types {#workflow-trigger-types} | ||||
|  | ||||
| Currently, there are three events that correspond to workflow trigger 'types': | ||||
|  | ||||
| @@ -379,9 +422,10 @@ Currently, there are three events that correspond to workflow trigger 'types': | ||||
|    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. | ||||
|    tags, doc type, correspondent or storage path. | ||||
| 4. **Scheduled**: a scheduled trigger that can be used to run workflows at a specific time. The date used can be either the document | ||||
|    added, created, updated date or you can specify a (date) custom field. You can also specify a day offset from the date. | ||||
|    added, created, updated date or you can specify a (date) custom field. You can also specify a day offset from the date (positive | ||||
|    offsets will trigger after the date, negative offsets will trigger before). | ||||
|  | ||||
| The following flow diagram illustrates the three document trigger types: | ||||
|  | ||||
| @@ -422,18 +466,19 @@ Workflows allow you to filter by: | ||||
| -   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 | ||||
| -   Content matching (`Added`, `Updated` and `Scheduled` triggers only). Filter document content using the matching settings. | ||||
| -   Tags (`Added`, `Updated` and `Scheduled` triggers only). Filter for documents with any of the specified tags | ||||
| -   Document type (`Added`, `Updated` and `Scheduled` triggers only). Filter documents with this doc type | ||||
| -   Correspondent (`Added`, `Updated` and `Scheduled` triggers only). Filter documents with this correspondent | ||||
| -   Storage path (`Added`, `Updated` and `Scheduled` triggers only). Filter documents with this storage path | ||||
|  | ||||
| ### Workflow Actions | ||||
|  | ||||
| #### Types | ||||
| #### Types {#workflow-action-types} | ||||
|  | ||||
| The following workflow action types are available: | ||||
|  | ||||
| ##### Assignment | ||||
| ##### Assignment {#workflow-action-assignment} | ||||
|  | ||||
| "Assignment" actions can assign: | ||||
|  | ||||
| @@ -443,7 +488,7 @@ The following workflow action types are available: | ||||
| -   View and / or edit permissions to users or groups | ||||
| -   Custom fields. Note that no value for the field will be set | ||||
|  | ||||
| ##### Removal | ||||
| ##### Removal {#workflow-action-removal} | ||||
|  | ||||
| "Removal" actions can remove either all of or specific sets of the following: | ||||
|  | ||||
| @@ -452,7 +497,7 @@ The following workflow action types are available: | ||||
| -   View and / or edit permissions | ||||
| -   Custom fields | ||||
|  | ||||
| ##### Email | ||||
| ##### Email {#workflow-action-email} | ||||
|  | ||||
| "Email" actions can send documents via email. This action requires a mail server to be [configured](configuration.md#email-sending). You can specify: | ||||
|  | ||||
| @@ -460,7 +505,7 @@ The following workflow action types are available: | ||||
| -   The subject and body of the email, which can include placeholders, see [placeholders](usage.md#workflow-placeholders) below | ||||
| -   Whether to include the document as an attachment | ||||
|  | ||||
| ##### Webhook | ||||
| ##### Webhook {#workflow-action-webhook} | ||||
|  | ||||
| "Webhook" actions send a POST request to a specified URL. You can specify: | ||||
|  | ||||
| @@ -469,44 +514,65 @@ The following workflow action types are available: | ||||
| -   Encoding for the request body, either JSON or form data | ||||
| -   The request headers as key-value pairs | ||||
|  | ||||
| For security reasons, webhooks can be limited to specific ports and disallowed from connecting to local URLs. See the relevant | ||||
| [configuration settings](configuration.md#workflow-webhooks) to change this behavior. If you are allowing non-admins to create workflows, | ||||
| you may want to adjust these settings to prevent abuse. | ||||
|  | ||||
| #### Workflow placeholders | ||||
|  | ||||
| Some workflow text 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 text is to be set), no automatic tags etc. have been | ||||
| applied. You can use the following placeholders with any trigger type: | ||||
| Titles can be assigned by workflows using [Jinja templates](https://jinja.palletsprojects.com/en/3.1.x/templates/). | ||||
| This allows for complex logic to be used to generate the title, including [logical structures](https://jinja.palletsprojects.com/en/3.1.x/templates/#list-of-control-structures) | ||||
| and [filters](https://jinja.palletsprojects.com/en/3.1.x/templates/#id11). | ||||
| The template is provided as a string. | ||||
|  | ||||
| -   `{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 | ||||
| -   `{filename}`: current file name without extension | ||||
| Using Jinja2 Templates is also useful for [Date localization](advanced_usage.md#Date-Localization) in the title. | ||||
|  | ||||
| The available inputs differ depending on the type of workflow trigger. | ||||
| This is because at the time of consumption (when the text is to be set), no automatic tags etc. have been | ||||
| applied. You can use the following placeholders in the template 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 | ||||
| -   `{{filename}}`: current 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 | ||||
| -   `{doc_url}`: URL to the document in the web UI. Requires the `PAPERLESS_URL` setting to be set. | ||||
| -   `{{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 | ||||
| -   `{{doc_url}}`: URL to the document in the web UI. Requires the `PAPERLESS_URL` setting to be set. | ||||
|  | ||||
| ##### Examples | ||||
|  | ||||
| ```jinja2 | ||||
| {{ created | localize_date('MMMM', 'en_US') }} | ||||
| <!-- Output: "January" --> | ||||
|  | ||||
| {{ added | localize_date('MMMM', 'de_DE') }} | ||||
| <!-- Output: "Juni" --> # codespell:ignore | ||||
| ``` | ||||
|  | ||||
| ### 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. | ||||
| Given their potentially far-reaching capabilities, including changing the permissions of existing documents, 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). | ||||
| @@ -544,27 +610,16 @@ The following custom field types are supported: | ||||
| -   `Document Link`: reference(s) to other document(s) displayed as links, automatically creates a symmetrical link in reverse | ||||
| -   `Select`: a pre-defined list of strings from which the user can choose | ||||
|  | ||||
| ## 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 four basic editing operations for PDFs (these operations currently cannot be performed on non-PDF files): | ||||
| Paperless-ngx supports basic editing operations for PDFs (these operations currently cannot be performed on non-PDF files). When viewing an individual document you can | ||||
| open the 'PDF Editor' to use a simple UI for re-arranging, rotating, deleting pages and splitting documents. | ||||
|  | ||||
| -   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. | ||||
| -   Deleting pages: available from an individual document's details page. | ||||
| -   Rotating documents: available when selecting multiple documents for 'bulk editing' and via the pdf editor on an individual document's details page. | ||||
| -   Splitting documents: via the pdf editor on an individual document's details page. | ||||
| -   Deleting pages: via the pdf editor on an individual document's details page. | ||||
| -   Re-arranging pages: via the pdf editor on an individual document's details page. | ||||
|  | ||||
| !!! important | ||||
|  | ||||
|   | ||||
| @@ -52,12 +52,12 @@ if ! command -v wget &> /dev/null ; then | ||||
| fi | ||||
|  | ||||
| if ! command -v docker &> /dev/null ; then | ||||
| 	echo "docker executable not found. Is docker installed?" | ||||
| 	echo "docker executable not found. Is Docker installed?" | ||||
| 	exit 1 | ||||
| fi | ||||
|  | ||||
| if ! docker compose &> /dev/null ; then | ||||
| 	echo "docker compose plugin not found. Is docker compose installed?" | ||||
| 	echo "docker compose plugin not found. Is Docker Compose installed?" | ||||
| 	exit 1 | ||||
| fi | ||||
|  | ||||
| @@ -66,7 +66,7 @@ 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 (may require restarting shell)." | ||||
| 	echo "WARN: Use 'sudo usermod -aG docker $USER' to assign Docker permissions to the user (may require restarting the shell)." | ||||
| 	echo "" | ||||
| 	sleep 3 | ||||
| fi | ||||
| @@ -135,7 +135,7 @@ DATABASE_BACKEND=$ask_result | ||||
|  | ||||
| echo "" | ||||
| echo "Paperless is able to use Apache Tika to support Office documents such as" | ||||
| echo "Word, Excel, Powerpoint, and Libreoffice equivalents. This feature" | ||||
| echo "Word, Excel, PowerPoint, and LibreOffice equivalents. This feature" | ||||
| echo "requires more resources due to the required services." | ||||
| echo "" | ||||
|  | ||||
| @@ -157,7 +157,7 @@ echo "" | ||||
| echo "Specify the user id and group id you wish to run paperless as." | ||||
| echo "Paperless will also change ownership on the data, media and consume" | ||||
| echo "folder to the specified values, so it's a good idea to supply the user id" | ||||
| echo "and group id of your unix user account." | ||||
| echo "and group id of your Unix user account." | ||||
| echo "If unsure, leave default." | ||||
| echo "" | ||||
|  | ||||
| @@ -212,7 +212,7 @@ if [[ "$DATABASE_BACKEND" == "sqlite" ]] ; then | ||||
| 	echo -n "SQLite database, the " | ||||
| fi | ||||
| echo "search index and other data." | ||||
| echo "As with the media folder, leave empty to have this managed by docker." | ||||
| echo "As with the media folder, leave empty to have this managed by Docker." | ||||
| echo "" | ||||
| echo "CAUTION: If specified, you must specify an absolute path starting with /" | ||||
| echo "or a relative path starting with ./ here." | ||||
| @@ -224,7 +224,7 @@ DATA_FOLDER=$ask_result | ||||
| if [[ "$DATABASE_BACKEND" == "postgres" || "$DATABASE_BACKEND" == "mariadb" ]] ; then | ||||
| 	echo "" | ||||
| 	echo "The database folder, where your database stores its data." | ||||
| 	echo "Leave empty to have this managed by docker." | ||||
| 	echo "Leave empty to have this managed by Docker." | ||||
| 	echo "" | ||||
| 	echo "CAUTION: If specified, you must specify an absolute path starting with /" | ||||
| 	echo "or a relative path starting with ./ here." | ||||
| @@ -276,18 +276,18 @@ echo "" | ||||
| echo "Target folder: $TARGET_FOLDER" | ||||
| echo "Consume folder: $CONSUME_FOLDER" | ||||
| if [[ -z $MEDIA_FOLDER ]] ; then | ||||
| 	echo "Media folder: Managed by docker" | ||||
| 	echo "Media folder: Managed by Docker" | ||||
| else | ||||
| 	echo "Media folder: $MEDIA_FOLDER" | ||||
| fi | ||||
| if [[ -z $DATA_FOLDER ]] ; then | ||||
| 	echo "Data folder: Managed by docker" | ||||
| 	echo "Data folder: Managed by Docker" | ||||
| else | ||||
| 	echo "Data folder: $DATA_FOLDER" | ||||
| fi | ||||
| if [[ "$DATABASE_BACKEND" == "postgres" || "$DATABASE_BACKEND" == "mariadb" ]] ; then | ||||
| 	if [[ -z $DATABASE_FOLDER ]] ; then | ||||
| 		echo "Database folder: Managed by docker" | ||||
| 		echo "Database folder: Managed by Docker" | ||||
| 	else | ||||
| 		echo "Database folder: $DATABASE_FOLDER" | ||||
| 	fi | ||||
|   | ||||
							
								
								
									
										29
									
								
								mkdocs.yml
									
									
									
									
									
								
							
							
						
						
									
										29
									
								
								mkdocs.yml
									
									
									
									
									
								
							| @@ -11,14 +11,12 @@ theme: | ||||
|       toggle: | ||||
|         icon: material/brightness-auto | ||||
|         name: Switch to light mode | ||||
|  | ||||
|     # Palette toggle for light mode | ||||
|     - media: "(prefers-color-scheme: light)" | ||||
|       scheme: default | ||||
|       toggle: | ||||
|         icon: material/brightness-7 | ||||
|         name: Switch to dark mode | ||||
|  | ||||
|     # Palette toggle for dark mode | ||||
|     - media: "(prefers-color-scheme: dark)" | ||||
|       scheme: slate | ||||
| @@ -49,6 +47,7 @@ markdown_extensions: | ||||
|   - pymdownx.superfences | ||||
|   - pymdownx.inlinehilite | ||||
|   - pymdownx.snippets | ||||
|   - pymdownx.tilde | ||||
|   - footnotes | ||||
|   - pymdownx.superfences: | ||||
|       custom_fences: | ||||
| @@ -60,17 +59,17 @@ markdown_extensions: | ||||
|       emoji_generator: !!python/name:material.extensions.emoji.to_svg | ||||
| strict: true | ||||
| nav: | ||||
|     - index.md | ||||
|     - setup.md | ||||
|     - 'Basic Usage': usage.md | ||||
|     - configuration.md | ||||
|     - administration.md | ||||
|     - advanced_usage.md | ||||
|     - 'REST API': api.md | ||||
|     - development.md | ||||
|     - 'FAQs': faq.md | ||||
|     - troubleshooting.md | ||||
|     - changelog.md | ||||
|   - index.md | ||||
|   - setup.md | ||||
|   - 'Basic Usage': usage.md | ||||
|   - configuration.md | ||||
|   - administration.md | ||||
|   - advanced_usage.md | ||||
|   - 'REST API': api.md | ||||
|   - development.md | ||||
|   - 'FAQs': faq.md | ||||
|   - troubleshooting.md | ||||
|   - changelog.md | ||||
| copyright: Copyright © 2016 - 2023 Daniel Quinn, Jonas Winkler, and the Paperless-ngx team | ||||
| extra: | ||||
|   social: | ||||
| @@ -83,5 +82,5 @@ extra: | ||||
| plugins: | ||||
|   - search | ||||
|   - glightbox: | ||||
|        skip_classes: | ||||
|          - no-lightbox | ||||
|       skip_classes: | ||||
|         - no-lightbox | ||||
|   | ||||
| @@ -1,10 +1,6 @@ | ||||
| # Have a look at the docs for documentation. | ||||
| # https://docs.paperless-ngx.com/configuration/ | ||||
|  | ||||
| # Debug. Only enable this for development. | ||||
|  | ||||
| #PAPERLESS_DEBUG=false | ||||
|  | ||||
| # Required services | ||||
|  | ||||
| #PAPERLESS_REDIS=redis://localhost:6379 | ||||
|   | ||||
							
								
								
									
										148
									
								
								pyproject.toml
									
									
									
									
									
								
							
							
						
						
									
										148
									
								
								pyproject.toml
									
									
									
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
| [project] | ||||
| name = "paperless-ngx" | ||||
| version = "2.15.0" | ||||
| description = "A community-supported supercharged version of paperless: scan, index and archive all your physical documents" | ||||
| version = "2.18.4" | ||||
| description = "A community-supported supercharged document management system: scan, index and archive all your physical documents" | ||||
| readme = "README.md" | ||||
| requires-python = ">=3.10" | ||||
| classifiers = [ | ||||
| @@ -15,57 +15,61 @@ classifiers = [ | ||||
| # This will allow testing to not install a webserver, mysql, etc | ||||
|  | ||||
| dependencies = [ | ||||
|   "babel>=2.17", | ||||
|   "bleach~=6.2.0", | ||||
|   "celery[redis]~=5.4.0", | ||||
|   "celery[redis]~=5.5.1", | ||||
|   "channels~=4.2", | ||||
|   "channels-redis~=4.2", | ||||
|   "concurrent-log-handler~=0.9.25", | ||||
|   "dateparser~=1.2", | ||||
|   # WARNING: django does not use semver. | ||||
|   #          Only patch versions are guaranteed to not introduce breaking changes. | ||||
|   "django~=5.1.6", | ||||
|   "django~=5.2.5", | ||||
|   "django-allauth[socialaccount,mfa]~=65.4.0", | ||||
|   "django-auditlog~=3.0.0", | ||||
|   "django-celery-results~=2.5.1", | ||||
|   "django-auditlog~=3.2.1", | ||||
|   "django-cachalot~=2.8.0", | ||||
|   "django-celery-results~=2.6.0", | ||||
|   "django-compression-middleware~=0.5.0", | ||||
|   "django-cors-headers~=4.7.0", | ||||
|   "django-extensions~=3.2.3", | ||||
|   "django-cors-headers~=4.8.0", | ||||
|   "django-extensions~=4.1", | ||||
|   "django-filter~=25.1", | ||||
|   "django-guardian~=2.4.0", | ||||
|   "django-multiselectfield~=0.1.13", | ||||
|   "django-guardian~=3.1.2", | ||||
|   "django-multiselectfield~=1.0.1", | ||||
|   "django-soft-delete~=1.0.18", | ||||
|   "djangorestframework~=3.15", | ||||
|   "djangorestframework-guardian~=0.3.0", | ||||
|   "django-treenode>=0.23.2", | ||||
|   "djangorestframework~=3.16", | ||||
|   "djangorestframework-guardian~=0.4.0", | ||||
|   "drf-spectacular~=0.28", | ||||
|   "drf-spectacular-sidecar~=2025.3.1", | ||||
|   "drf-spectacular-sidecar~=2025.9.1", | ||||
|   "drf-writable-nested~=0.7.1", | ||||
|   "filelock~=3.17.0", | ||||
|   "filelock~=3.19.1", | ||||
|   "flower~=2.0.1", | ||||
|   "gotenberg-client~=0.9.0", | ||||
|   "gotenberg-client~=0.11.0", | ||||
|   "httpx-oauth~=0.16", | ||||
|   "imap-tools~=1.10.0", | ||||
|   "imap-tools~=1.11.0", | ||||
|   "inotifyrecursive~=0.3", | ||||
|   "jinja2~=3.1.5", | ||||
|   "langdetect~=1.0.9", | ||||
|   "nltk~=3.9.1", | ||||
|   "ocrmypdf~=16.10.0", | ||||
|   "pathvalidate~=3.2.3", | ||||
|   "ocrmypdf~=16.11.0", | ||||
|   "pathvalidate~=3.3.1", | ||||
|   "pdf2image~=1.17.0", | ||||
|   "psycopg-pool", | ||||
|   "python-dateutil~=2.9.0", | ||||
|   "python-dotenv~=1.0.1", | ||||
|   "python-dotenv~=1.1.0", | ||||
|   "python-gnupg~=0.5.4", | ||||
|   "python-ipware~=3.0.0", | ||||
|   "python-magic~=0.4.27", | ||||
|   "pyzbar~=0.1.9", | ||||
|   "rapidfuzz~=3.12.1", | ||||
|   "rapidfuzz~=3.14.0", | ||||
|   "redis[hiredis]~=5.2.1", | ||||
|   "scikit-learn~=1.6.1", | ||||
|   "scikit-learn~=1.7.0", | ||||
|   "setproctitle~=1.3.4", | ||||
|   "tika-client~=0.9.0", | ||||
|   "tika-client~=0.10.0", | ||||
|   "tqdm~=4.67.1", | ||||
|   "watchdog~=6.0", | ||||
|   "whitenoise~=6.9", | ||||
|   "whoosh~=2.7", | ||||
|   "whoosh-reloaded>=2.7.5", | ||||
|   "zxing-cpp~=2.3.0", | ||||
| ] | ||||
|  | ||||
| @@ -73,12 +77,13 @@ optional-dependencies.mariadb = [ | ||||
|   "mysqlclient~=2.2.7", | ||||
| ] | ||||
| optional-dependencies.postgres = [ | ||||
|   "psycopg[c]==3.2.5", | ||||
|   "psycopg[c,pool]==3.2.9", | ||||
|   # Direct dependency for proper resolution of the pre-built wheels | ||||
|   "psycopg-c==3.2.5", | ||||
|   "psycopg-c==3.2.9", | ||||
|   "psycopg-pool==3.2.6", | ||||
| ] | ||||
| optional-dependencies.webserver = [ | ||||
|   "granian~=2.0.1", | ||||
|   "granian[uvloop]~=2.5.1", | ||||
| ] | ||||
|  | ||||
| [dependency-groups] | ||||
| @@ -90,7 +95,7 @@ dev = [ | ||||
| ] | ||||
|  | ||||
| docs = [ | ||||
|   "mkdocs-glightbox~=0.4.0", | ||||
|   "mkdocs-glightbox~=0.5.1", | ||||
|   "mkdocs-material~=9.6.4", | ||||
| ] | ||||
|  | ||||
| @@ -98,9 +103,9 @@ testing = [ | ||||
|   "daphne", | ||||
|   "factory-boy~=3.3.1", | ||||
|   "imagehash", | ||||
|   "pytest~=8.3.3", | ||||
|   "pytest-cov~=6.0.0", | ||||
|   "pytest-django~=4.10.0", | ||||
|   "pytest~=8.4.1", | ||||
|   "pytest-cov~=7.0.0", | ||||
|   "pytest-django~=4.11.1", | ||||
|   "pytest-env", | ||||
|   "pytest-httpx", | ||||
|   "pytest-mock", | ||||
| @@ -110,9 +115,9 @@ testing = [ | ||||
| ] | ||||
|  | ||||
| lint = [ | ||||
|   "pre-commit~=4.1.0", | ||||
|   "pre-commit~=4.3.0", | ||||
|   "pre-commit-uv~=4.1.3", | ||||
|   "ruff~=0.9.9", | ||||
|   "ruff~=0.13.0", | ||||
| ] | ||||
|  | ||||
| typing = [ | ||||
| @@ -120,6 +125,7 @@ typing = [ | ||||
|   "django-filter-stubs", | ||||
|   "django-stubs[compatible-mypy]", | ||||
|   "djangorestframework-stubs[compatible-mypy]", | ||||
|   "lxml-stubs", | ||||
|   "mypy", | ||||
|   "types-bleach", | ||||
|   "types-colorama", | ||||
| @@ -127,6 +133,7 @@ typing = [ | ||||
|   "types-markdown", | ||||
|   "types-pygments", | ||||
|   "types-python-dateutil", | ||||
|   "types-pytz", | ||||
|   "types-redis", | ||||
|   "types-setuptools", | ||||
|   "types-tqdm", | ||||
| @@ -172,6 +179,7 @@ lint.extend-select = [ | ||||
| ] | ||||
| lint.ignore = [ | ||||
|   "DJ001", | ||||
|   "PLC0415", | ||||
|   "RUF012", | ||||
|   "SIM105", | ||||
| ] | ||||
| @@ -200,75 +208,19 @@ lint.per-file-ignores."docker/wait-for-redis.py" = [ | ||||
|   "INP001", | ||||
|   "T201", | ||||
| ] | ||||
| lint.per-file-ignores."src/documents/file_handling.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/management/commands/document_consumer.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/management/commands/document_exporter.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/migrations/1012_fix_archive_files.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/models.py" = [ | ||||
|   "SIM115", | ||||
| ] | ||||
| lint.per-file-ignores."src/documents/parsers.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/signals/handlers.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_consumer.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_file_handling.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_management.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_management_consumer.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_management_exporter.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_migration_archive_files.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_migration_document_pages_count.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_migration_mime_type.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/tests/test_sanity_check.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/documents/views.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/paperless/checks.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/paperless/settings.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/paperless/views.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/paperless_mail/mail.py" = [ | ||||
|   "PTH", | ||||
| ] # TODO Enable & remove | ||||
| lint.per-file-ignores."src/paperless_tesseract/tests/test_parser.py" = [ | ||||
|   "PTH", | ||||
|   "RUF001", | ||||
| ] # TODO PTH Enable & remove | ||||
| ] | ||||
| lint.isort.force-single-line = true | ||||
|  | ||||
| [tool.codespell] | ||||
| write-changes = true | ||||
| ignore-words-list = "criterias,afterall,valeu,ureue,equest,ure,assertIn,Oktober" | ||||
| skip = "src-ui/src/locale/*,src-ui/pnpm-lock.yaml,src-ui/e2e/*,src/paperless_mail/tests/samples/*,src/documents/tests/samples/*,*.po,*.json" | ||||
|  | ||||
| [tool.pytest.ini_options] | ||||
| minversion = "8.0" | ||||
| pythonpath = [ | ||||
| @@ -280,6 +232,7 @@ testpaths = [ | ||||
|   "src/paperless_mail/tests/", | ||||
|   "src/paperless_tesseract/tests/", | ||||
|   "src/paperless_tika/tests", | ||||
|   "src/paperless_text/tests/", | ||||
| ] | ||||
| addopts = [ | ||||
|   "--pythonwarnings=all", | ||||
| @@ -302,6 +255,7 @@ PAPERLESS_DISABLE_DBHANDLER = "true" | ||||
| PAPERLESS_CACHE_BACKEND = "django.core.cache.backends.locmem.LocMemCache" | ||||
|  | ||||
| [tool.coverage.run] | ||||
| relative_files = true | ||||
| source = [ | ||||
|   "src/", | ||||
| ] | ||||
| @@ -320,10 +274,10 @@ exclude_also = [ | ||||
| ] | ||||
|  | ||||
| [tool.mypy] | ||||
| mypy_path = "src" | ||||
| plugins = [ | ||||
|   "mypy_django_plugin.main", | ||||
|   "mypy_drf_plugin.main", | ||||
|   "numpy.typing.mypy_plugin", | ||||
| ] | ||||
| check_untyped_defs = true | ||||
| disallow_any_generics = true | ||||
| @@ -343,8 +297,8 @@ environments = [ | ||||
| [tool.uv.sources] | ||||
| # Markers are chosen to select these almost exclusively when building the Docker image | ||||
| psycopg-c = [ | ||||
|   { url = "https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.5/psycopg_c-3.2.5-cp312-cp312-linux_x86_64.whl", marker = "sys_platform == 'linux' and platform_machine == 'x86_64' and python_version == '3.12'" }, | ||||
|   { url = "https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.5/psycopg_c-3.2.5-cp312-cp312-linux_aarch64.whl", marker = "sys_platform == 'linux' and platform_machine == 'aarch64' and python_version == '3.12'" }, | ||||
|   { url = "https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.9/psycopg_c-3.2.9-cp312-cp312-linux_x86_64.whl", marker = "sys_platform == 'linux' and platform_machine == 'x86_64' and python_version == '3.12'" }, | ||||
|   { url = "https://github.com/paperless-ngx/builder/releases/download/psycopg-3.2.9/psycopg_c-3.2.9-cp312-cp312-linux_aarch64.whl", marker = "sys_platform == 'linux' and platform_machine == 'aarch64' and python_version == '3.12'" }, | ||||
| ] | ||||
| zxing-cpp = [ | ||||
|   { url = "https://github.com/paperless-ngx/builder/releases/download/zxing-2.3.0/zxing_cpp-2.3.0-cp312-cp312-linux_x86_64.whl", marker = "sys_platform == 'linux' and platform_machine == 'x86_64' and python_version == '3.12'" }, | ||||
|   | ||||
| @@ -6,6 +6,7 @@ A document with an id of ${DOCUMENT_ID} was just consumed.  I know the | ||||
| following additional information about it: | ||||
|  | ||||
| * Generated File Name: ${DOCUMENT_FILE_NAME} | ||||
| * Document type: ${DOCUMENT_TYPE} | ||||
| * Archive Path: ${DOCUMENT_ARCHIVE_PATH} | ||||
| * Source Path: ${DOCUMENT_SOURCE_PATH} | ||||
| * Created: ${DOCUMENT_CREATED} | ||||
|   | ||||
							
								
								
									
										24
									
								
								sonar-project.properties
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								sonar-project.properties
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | ||||
| sonar.projectKey=paperless-ngx_paperless-ngx | ||||
| sonar.organization=paperless-ngx | ||||
| sonar.projectName=Paperless-ngx | ||||
| sonar.projectVersion=1.0 | ||||
|  | ||||
| # Source and test directories | ||||
| sonar.sources=src/,src-ui/ | ||||
| sonar.test.inclusions=**/test_*.py,**/tests.py,**/*.spec.ts,**/*.test.ts | ||||
|  | ||||
| # Language specific settings | ||||
| sonar.python.version=3.10,3.11,3.12,3.13 | ||||
|  | ||||
| # Coverage reports | ||||
| sonar.python.coverage.reportPaths=merged-backend-coverage.xml | ||||
| sonar.javascript.lcov.reportPaths=coverage/lcov.info | ||||
|  | ||||
| # Test execution reports | ||||
| sonar.junit.reportPaths=**/junit.xml,**/test-results.xml | ||||
|  | ||||
| # Encoding | ||||
| sonar.sourceEncoding=UTF-8 | ||||
|  | ||||
| # Exclusions | ||||
| sonar.exclusions=**/migrations/**,**/node_modules/**,**/static/**,**/venv/**,**/.venv/**,**/dist/** | ||||
							
								
								
									
										13
									
								
								src-ui/__mocks__/pdfjs-dist.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								src-ui/__mocks__/pdfjs-dist.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| export const getDocument = jest.fn(() => ({ | ||||
|   promise: Promise.resolve({ numPages: 3 }), | ||||
| })) | ||||
|  | ||||
| export const GlobalWorkerOptions = { workerSrc: '' } | ||||
| export const VerbosityLevel = { ERRORS: 0 } | ||||
|  | ||||
| globalThis.pdfjsLib = { | ||||
|   getDocument, | ||||
|   GlobalWorkerOptions, | ||||
|   VerbosityLevel, | ||||
|   AbortException: class AbortException extends Error {}, | ||||
| } | ||||
| @@ -27,6 +27,7 @@ | ||||
|           "el-GR": "src/locale/messages.el_GR.xlf", | ||||
|           "en-GB": "src/locale/messages.en_GB.xlf", | ||||
|           "es-ES": "src/locale/messages.es_ES.xlf", | ||||
|           "fa-IR": "src/locale/messages.fa_IR.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", | ||||
| @@ -47,6 +48,7 @@ | ||||
|           "sv-SE": "src/locale/messages.sv_SE.xlf", | ||||
|           "tr-TR": "src/locale/messages.tr_TR.xlf", | ||||
|           "uk-UA": "src/locale/messages.uk_UA.xlf", | ||||
|           "vi-VN": "src/locale/messages.vi_VN.xlf", | ||||
|           "zh-CN": "src/locale/messages.zh_CN.xlf", | ||||
|           "zh-TW": "src/locale/messages.zh_TW.xlf" | ||||
|         } | ||||
| @@ -59,10 +61,12 @@ | ||||
|                 "path": "./extra-webpack.config.ts" | ||||
|             }, | ||||
|             "outputPath": "dist/paperless-ui", | ||||
|             "main": "src/main.ts", | ||||
|             "outputHashing": "none", | ||||
|             "index": "src/index.html", | ||||
|             "main": "src/main.ts", | ||||
|             "polyfills": "src/polyfills.ts", | ||||
|             "polyfills": [ | ||||
|               "src/polyfills.ts" | ||||
|             ], | ||||
|             "tsConfig": "tsconfig.app.json", | ||||
|             "localize": true, | ||||
|             "assets": [ | ||||
| @@ -85,12 +89,15 @@ | ||||
|               "file-saver", | ||||
|               "utif" | ||||
|             ], | ||||
|             "vendorChunk": true, | ||||
|             "extractLicenses": false, | ||||
|             "buildOptimizer": false, | ||||
|             "sourceMap": true, | ||||
|             "optimization": false, | ||||
|             "namedChunks": true | ||||
|             "namedChunks": true, | ||||
|             "stylePreprocessorOptions": { | ||||
|               "includePaths": [ | ||||
|                 "." | ||||
|               ] | ||||
|             } | ||||
|           }, | ||||
|           "configurations": { | ||||
|             "production": { | ||||
| @@ -106,8 +113,6 @@ | ||||
|               "sourceMap": false, | ||||
|               "namedChunks": false, | ||||
|               "extractLicenses": true, | ||||
|               "vendorChunk": false, | ||||
|               "buildOptimizer": true, | ||||
|               "budgets": [ | ||||
|                 { | ||||
|                   "type": "initial", | ||||
| @@ -187,6 +192,30 @@ | ||||
|     }, | ||||
|     "@angular-eslint/schematics:library": { | ||||
|       "setParserOptionsProject": true | ||||
|     }, | ||||
|     "@schematics/angular:component": { | ||||
|       "type": "component" | ||||
|     }, | ||||
|     "@schematics/angular:directive": { | ||||
|       "type": "directive" | ||||
|     }, | ||||
|     "@schematics/angular:service": { | ||||
|       "type": "service" | ||||
|     }, | ||||
|     "@schematics/angular:guard": { | ||||
|       "typeSeparator": "." | ||||
|     }, | ||||
|     "@schematics/angular:interceptor": { | ||||
|       "typeSeparator": "." | ||||
|     }, | ||||
|     "@schematics/angular:module": { | ||||
|       "typeSeparator": "." | ||||
|     }, | ||||
|     "@schematics/angular:pipe": { | ||||
|       "typeSeparator": "." | ||||
|     }, | ||||
|     "@schematics/angular:resolver": { | ||||
|       "typeSeparator": "." | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,7 @@ module.exports = { | ||||
|     'abstract-paperless-service', | ||||
|   ], | ||||
|   transformIgnorePatterns: [ | ||||
|     `<rootDir>/node_modules/.pnpm/(?!.*\\.mjs$|lodash-es)`, | ||||
|     `<rootDir>/node_modules/.pnpm/(?!.*\\.mjs$|lodash-es|@angular\\+common.*locales)`, | ||||
|   ], | ||||
|   moduleNameMapper: { | ||||
|     '^src/(.*)': '<rootDir>/src/$1', | ||||
|   | ||||
							
								
								
									
										2318
									
								
								src-ui/messages.xlf
									
									
									
									
									
								
							
							
						
						
									
										2318
									
								
								src-ui/messages.xlf
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,76 +1,74 @@ | ||||
| { | ||||
|   "name": "paperless-ui", | ||||
|   "version": "0.0.0", | ||||
|   "name": "paperless-ngx-ui", | ||||
|   "version": "2.18.4", | ||||
|   "scripts": { | ||||
|     "preinstall": "npx only-allow pnpm", | ||||
|     "ng": "ng", | ||||
|     "start": "ng serve", | ||||
|     "build": "ng build", | ||||
|     "test": "ng test --no-watch --coverage", | ||||
|     "lint": "ng lint", | ||||
|     "postinstall": "patch-package" | ||||
|     "lint": "ng lint" | ||||
|   }, | ||||
|   "private": true, | ||||
|   "dependencies": { | ||||
|     "@angular/cdk": "^19.2.2", | ||||
|     "@angular/common": "~19.2.1", | ||||
|     "@angular/compiler": "~19.2.1", | ||||
|     "@angular/core": "~19.2.1", | ||||
|     "@angular/forms": "~19.2.1", | ||||
|     "@angular/localize": "~19.2.1", | ||||
|     "@angular/platform-browser": "~19.2.1", | ||||
|     "@angular/platform-browser-dynamic": "~19.2.1", | ||||
|     "@angular/router": "~19.2.1", | ||||
|     "@ng-bootstrap/ng-bootstrap": "^18.0.0", | ||||
|     "@ng-select/ng-select": "^14.2.3", | ||||
|     "@angular/cdk": "^20.2.2", | ||||
|     "@angular/common": "~20.2.4", | ||||
|     "@angular/compiler": "~20.2.4", | ||||
|     "@angular/core": "~20.2.4", | ||||
|     "@angular/forms": "~20.2.4", | ||||
|     "@angular/localize": "~20.2.4", | ||||
|     "@angular/platform-browser": "~20.2.4", | ||||
|     "@angular/platform-browser-dynamic": "~20.2.4", | ||||
|     "@angular/router": "~20.2.4", | ||||
|     "@ng-bootstrap/ng-bootstrap": "^19.0.1", | ||||
|     "@ng-select/ng-select": "^20.1.3", | ||||
|     "@ngneat/dirty-check-forms": "^3.0.3", | ||||
|     "@popperjs/core": "^2.11.8", | ||||
|     "bootstrap": "^5.3.3", | ||||
|     "bootstrap": "^5.3.8", | ||||
|     "file-saver": "^2.0.5", | ||||
|     "mime-names": "^1.0.0", | ||||
|     "ng2-pdf-viewer": "^10.4.0", | ||||
|     "ngx-bootstrap-icons": "^1.9.3", | ||||
|     "ngx-color": "^10.0.0", | ||||
|     "ngx-cookie-service": "^19.1.2", | ||||
|     "ngx-device-detector": "^9.0.0", | ||||
|     "ngx-file-drop": "^16.0.0", | ||||
|     "ngx-ui-tour-ng-bootstrap": "^16.0.0", | ||||
|     "ngx-cookie-service": "^20.1.0", | ||||
|     "ngx-device-detector": "^10.1.0", | ||||
|     "ngx-ui-tour-ng-bootstrap": "^17.0.1", | ||||
|     "rxjs": "^7.8.2", | ||||
|     "tslib": "^2.8.1", | ||||
|     "utif": "^3.1.0", | ||||
|     "uuid": "^11.1.0", | ||||
|     "zone.js": "^0.15.0" | ||||
|     "zone.js": "^0.15.1" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@angular-builders/custom-webpack": "^19.0.0", | ||||
|     "@angular-builders/jest": "^19.0.0", | ||||
|     "@angular-devkit/build-angular": "^19.2.1", | ||||
|     "@angular-devkit/core": "^19.2.1", | ||||
|     "@angular-devkit/schematics": "^19.2.1", | ||||
|     "@angular-eslint/builder": "19.2.1", | ||||
|     "@angular-eslint/eslint-plugin": "19.2.1", | ||||
|     "@angular-eslint/eslint-plugin-template": "19.2.1", | ||||
|     "@angular-eslint/schematics": "19.2.1", | ||||
|     "@angular-eslint/template-parser": "19.2.1", | ||||
|     "@angular/cli": "~19.2.1", | ||||
|     "@angular/compiler-cli": "~19.2.1", | ||||
|     "@codecov/webpack-plugin": "^1.9.0", | ||||
|     "@playwright/test": "^1.50.1", | ||||
|     "@types/jest": "^29.5.14", | ||||
|     "@types/node": "^22.13.9", | ||||
|     "@typescript-eslint/eslint-plugin": "^8.26.1", | ||||
|     "@typescript-eslint/parser": "^8.26.1", | ||||
|     "@typescript-eslint/utils": "^8.26.1", | ||||
|     "eslint": "^9.22.0", | ||||
|     "jest": "29.7.0", | ||||
|     "jest-environment-jsdom": "^29.7.0", | ||||
|     "@angular-builders/custom-webpack": "^20.0.0", | ||||
|     "@angular-builders/jest": "^20.0.0", | ||||
|     "@angular-devkit/core": "^20.2.2", | ||||
|     "@angular-devkit/schematics": "^20.2.2", | ||||
|     "@angular-eslint/builder": "20.2.0", | ||||
|     "@angular-eslint/eslint-plugin": "20.2.0", | ||||
|     "@angular-eslint/eslint-plugin-template": "20.2.0", | ||||
|     "@angular-eslint/schematics": "20.2.0", | ||||
|     "@angular-eslint/template-parser": "20.2.0", | ||||
|     "@angular/build": "^20.2.2", | ||||
|     "@angular/cli": "~20.2.2", | ||||
|     "@angular/compiler-cli": "~20.2.4", | ||||
|     "@codecov/webpack-plugin": "^1.9.1", | ||||
|     "@playwright/test": "^1.55.0", | ||||
|     "@types/jest": "^30.0.0", | ||||
|     "@types/node": "^24.3.0", | ||||
|     "@typescript-eslint/eslint-plugin": "^8.41.0", | ||||
|     "@typescript-eslint/parser": "^8.41.0", | ||||
|     "@typescript-eslint/utils": "^8.41.0", | ||||
|     "eslint": "^9.34.0", | ||||
|     "jest": "30.1.3", | ||||
|     "jest-environment-jsdom": "^30.1.2", | ||||
|     "jest-junit": "^16.0.0", | ||||
|     "jest-preset-angular": "^14.5.3", | ||||
|     "jest-preset-angular": "^15.0.0", | ||||
|     "jest-websocket-mock": "^2.5.0", | ||||
|     "patch-package": "^8.0.0", | ||||
|     "prettier-plugin-organize-imports": "^4.1.0", | ||||
|     "prettier-plugin-organize-imports": "^4.2.0", | ||||
|     "ts-node": "~10.9.1", | ||||
|     "typescript": "^5.5.4" | ||||
|     "typescript": "^5.8.3", | ||||
|     "webpack": "^5.101.3" | ||||
|   }, | ||||
|   "pnpm": { | ||||
|     "onlyBuiltDependencies": [ | ||||
| @@ -80,6 +78,5 @@ | ||||
|       "lmdb", | ||||
|       "msgpackr-extract" | ||||
|     ] | ||||
|   }, | ||||
|   "typings": "./src/typings.d.ts" | ||||
|   } | ||||
| } | ||||
|   | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										9478
									
								
								src-ui/pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9478
									
								
								src-ui/pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,12 +1,16 @@ | ||||
| import '@angular/localize/init' | ||||
| import { jest } from '@jest/globals' | ||||
| import { setupZoneTestEnv } from 'jest-preset-angular/setup-env/zone' | ||||
| import { TextDecoder, TextEncoder } from 'util' | ||||
| import { TextDecoder, TextEncoder } from 'node:util' | ||||
| if (process.env.NODE_ENV === 'test') { | ||||
|   setupZoneTestEnv() | ||||
| } | ||||
| global.TextEncoder = TextEncoder | ||||
| global.TextDecoder = TextDecoder | ||||
| ;(globalThis as any).TextEncoder = TextEncoder as unknown as { | ||||
|   new (): TextEncoder | ||||
| } | ||||
| ;(globalThis as any).TextDecoder = TextDecoder as unknown as { | ||||
|   new (): TextDecoder | ||||
| } | ||||
|  | ||||
| import { registerLocaleData } from '@angular/common' | ||||
| import localeAf from '@angular/common/locales/af' | ||||
| @@ -20,6 +24,7 @@ import localeDe from '@angular/common/locales/de' | ||||
| import localeEl from '@angular/common/locales/el' | ||||
| import localeEnGb from '@angular/common/locales/en-GB' | ||||
| import localeEs from '@angular/common/locales/es' | ||||
| import localeFa from '@angular/common/locales/fa' | ||||
| import localeFi from '@angular/common/locales/fi' | ||||
| import localeFr from '@angular/common/locales/fr' | ||||
| import localeHu from '@angular/common/locales/hu' | ||||
| @@ -39,6 +44,7 @@ import localeSr from '@angular/common/locales/sr' | ||||
| import localeSv from '@angular/common/locales/sv' | ||||
| import localeTr from '@angular/common/locales/tr' | ||||
| import localeUk from '@angular/common/locales/uk' | ||||
| import localeVi from '@angular/common/locales/vi' | ||||
| import localeZh from '@angular/common/locales/zh' | ||||
| import localeZhHant from '@angular/common/locales/zh-Hant' | ||||
|  | ||||
| @@ -53,6 +59,7 @@ registerLocaleData(localeDe) | ||||
| registerLocaleData(localeEl) | ||||
| registerLocaleData(localeEnGb) | ||||
| registerLocaleData(localeEs) | ||||
| registerLocaleData(localeFa) | ||||
| registerLocaleData(localeFi) | ||||
| registerLocaleData(localeFr) | ||||
| registerLocaleData(localeHu) | ||||
| @@ -73,6 +80,7 @@ registerLocaleData(localeSr) | ||||
| registerLocaleData(localeSv) | ||||
| registerLocaleData(localeTr) | ||||
| registerLocaleData(localeUk) | ||||
| registerLocaleData(localeVi) | ||||
| registerLocaleData(localeZh) | ||||
| registerLocaleData(localeZhHant) | ||||
|  | ||||
| @@ -112,28 +120,29 @@ if (!URL.revokeObjectURL) { | ||||
|   Object.defineProperty(window.URL, 'revokeObjectURL', { value: jest.fn() }) | ||||
| } | ||||
| Object.defineProperty(window, 'ResizeObserver', { value: mock() }) | ||||
| Object.defineProperty(window, 'location', { | ||||
|   configurable: true, | ||||
|   value: { reload: jest.fn() }, | ||||
| }) | ||||
|  | ||||
| if (typeof IntersectionObserver === 'undefined') { | ||||
|   class MockIntersectionObserver { | ||||
|     constructor( | ||||
|       public callback: IntersectionObserverCallback, | ||||
|       public options?: IntersectionObserverInit | ||||
|     ) {} | ||||
|  | ||||
|     observe = jest.fn() | ||||
|     unobserve = jest.fn() | ||||
|     disconnect = jest.fn() | ||||
|     takeRecords = jest.fn() | ||||
|   } | ||||
|  | ||||
|   Object.defineProperty(window, 'IntersectionObserver', { | ||||
|     writable: true, | ||||
|     configurable: true, | ||||
|     value: MockIntersectionObserver, | ||||
|   }) | ||||
| } | ||||
|  | ||||
| HTMLCanvasElement.prototype.getContext = < | ||||
|   typeof HTMLCanvasElement.prototype.getContext | ||||
| >jest.fn() | ||||
|  | ||||
| // pdfjs | ||||
| jest.mock('pdfjs-dist', () => ({ | ||||
|   getDocument: jest.fn(() => ({ | ||||
|     promise: Promise.resolve({ numPages: 3 }), | ||||
|   })), | ||||
|   GlobalWorkerOptions: { workerSrc: '' }, | ||||
|   VerbosityLevel: { ERRORS: 0 }, | ||||
|   globalThis: { | ||||
|     pdfjsLib: { | ||||
|       GlobalWorkerOptions: { | ||||
|         workerSrc: '', | ||||
|       }, | ||||
|     }, | ||||
|   }, | ||||
| })) | ||||
| jest.mock('pdfjs-dist/web/pdf_viewer', () => ({})) | ||||
| jest.mock('pdfjs-dist') | ||||
|   | ||||
| @@ -9,7 +9,6 @@ import { | ||||
| import { Router, RouterModule } from '@angular/router' | ||||
| import { NgbModalModule } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { allIcons, NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { NgxFileDropModule } from 'ngx-file-drop' | ||||
| import { TourNgBootstrapModule, TourService } from 'ngx-ui-tour-ng-bootstrap' | ||||
| import { Subject } from 'rxjs' | ||||
| import { routes } from './app-routing.module' | ||||
| @@ -43,7 +42,6 @@ describe('AppComponent', () => { | ||||
|       imports: [ | ||||
|         TourNgBootstrapModule, | ||||
|         RouterModule.forRoot(routes), | ||||
|         NgxFileDropModule, | ||||
|         NgbModalModule, | ||||
|         AppComponent, | ||||
|         ToastsComponent, | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { Component, OnDestroy, OnInit, Renderer2 } from '@angular/core' | ||||
| import { Component, inject, OnDestroy, OnInit, Renderer2 } from '@angular/core' | ||||
| import { Router, RouterOutlet } from '@angular/router' | ||||
| import { TourNgBootstrapModule, TourService } from 'ngx-ui-tour-ng-bootstrap' | ||||
| import { first, Subscription } from 'rxjs' | ||||
| @@ -29,22 +29,22 @@ import { WebsocketStatusService } from './services/websocket-status.service' | ||||
|   ], | ||||
| }) | ||||
| export class AppComponent implements OnInit, OnDestroy { | ||||
|   private settings = inject(SettingsService) | ||||
|   private websocketStatusService = inject(WebsocketStatusService) | ||||
|   private toastService = inject(ToastService) | ||||
|   private router = inject(Router) | ||||
|   private tasksService = inject(TasksService) | ||||
|   tourService = inject(TourService) | ||||
|   private renderer = inject(Renderer2) | ||||
|   private permissionsService = inject(PermissionsService) | ||||
|   private hotKeyService = inject(HotKeyService) | ||||
|   private componentRouterService = inject(ComponentRouterService) | ||||
|  | ||||
|   newDocumentSubscription: Subscription | ||||
|   successSubscription: Subscription | ||||
|   failedSubscription: Subscription | ||||
|  | ||||
|   constructor( | ||||
|     private settings: SettingsService, | ||||
|     private websocketStatusService: WebsocketStatusService, | ||||
|     private toastService: ToastService, | ||||
|     private router: Router, | ||||
|     private tasksService: TasksService, | ||||
|     public tourService: TourService, | ||||
|     private renderer: Renderer2, | ||||
|     private permissionsService: PermissionsService, | ||||
|     private hotKeyService: HotKeyService, | ||||
|     private componentRouterService: ComponentRouterService | ||||
|   ) { | ||||
|   constructor() { | ||||
|     let anyWindow = window as any | ||||
|     anyWindow.pdfWorkerSrc = 'assets/js/pdf.worker.min.mjs' | ||||
|     this.settings.updateAppearanceSettings() | ||||
|   | ||||
| @@ -50,7 +50,7 @@ | ||||
|     <div [ngbNavOutlet]="nav" class="border-start border-end border-bottom p-3 mb-3 shadow-sm"></div> | ||||
|     <div class="btn-toolbar" role="toolbar"> | ||||
|         <div class="btn-group me-2"> | ||||
|             <button type="button" (click)="discardChanges()" class="btn btn-secondary" [disabled]="loading || (isDirty$ | async) === false" i18n>Discard</button> | ||||
|             <button type="button" (click)="discardChanges()" class="btn btn-outline-secondary" [disabled]="loading || (isDirty$ | async) === false" i18n>Discard</button> | ||||
|         </div> | ||||
|         <div class="btn-group"> | ||||
|             <button type="submit" class="btn btn-primary" [disabled]="loading || !configForm.valid || (isDirty$ | async) === false" i18n>Save</button> | ||||
|   | ||||
| @@ -105,9 +105,9 @@ describe('ConfigComponent', () => { | ||||
|  | ||||
|   it('should support JSON validation for e.g. user_args', () => { | ||||
|     component.configForm.patchValue({ user_args: '{ foo bar }' }) | ||||
|     expect(component.errors).toEqual({ user_args: 'Invalid JSON' }) | ||||
|     expect(component.errors['user_args']).toEqual('Invalid JSON') | ||||
|     component.configForm.patchValue({ user_args: '{ "foo": "bar" }' }) | ||||
|     expect(component.errors).toEqual({ user_args: null }) | ||||
|     expect(component.errors['user_args']).toBeNull() | ||||
|   }) | ||||
|  | ||||
|   it('should upload file, show error if necessary', () => { | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { AsyncPipe } from '@angular/common' | ||||
| import { Component, OnDestroy, OnInit } from '@angular/core' | ||||
| import { Component, OnDestroy, OnInit, inject } from '@angular/core' | ||||
| import { | ||||
|   AbstractControl, | ||||
|   FormControl, | ||||
| @@ -57,6 +57,10 @@ export class ConfigComponent | ||||
|   extends LoadingComponentWithPermissions | ||||
|   implements OnInit, OnDestroy, DirtyComponent | ||||
| { | ||||
|   private configService = inject(ConfigService) | ||||
|   private toastService = inject(ToastService) | ||||
|   private settingsService = inject(SettingsService) | ||||
|  | ||||
|   public readonly ConfigOptionType = ConfigOptionType | ||||
|  | ||||
|   // generated dynamically | ||||
| @@ -77,11 +81,7 @@ export class ConfigComponent | ||||
|   storeSub: Subscription | ||||
|   isDirty$: Observable<boolean> | ||||
|  | ||||
|   constructor( | ||||
|     private configService: ConfigService, | ||||
|     private toastService: ToastService, | ||||
|     private settingsService: SettingsService | ||||
|   ) { | ||||
|   constructor() { | ||||
|     super() | ||||
|     this.configForm.addControl('id', new FormControl()) | ||||
|     PaperlessConfigOptions.forEach((option) => { | ||||
|   | ||||
| @@ -5,6 +5,7 @@ import { | ||||
|   OnDestroy, | ||||
|   OnInit, | ||||
|   ViewChild, | ||||
|   inject, | ||||
| } from '@angular/core' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbNavModule } from '@ng-bootstrap/ng-bootstrap' | ||||
| @@ -28,12 +29,8 @@ export class LogsComponent | ||||
|   extends LoadingComponentWithPermissions | ||||
|   implements OnInit, OnDestroy | ||||
| { | ||||
|   constructor( | ||||
|     private logService: LogService, | ||||
|     private changedetectorRef: ChangeDetectorRef | ||||
|   ) { | ||||
|     super() | ||||
|   } | ||||
|   private logService = inject(LogService) | ||||
|   private changedetectorRef = inject(ChangeDetectorRef) | ||||
|  | ||||
|   public logs: string[] = [] | ||||
|  | ||||
|   | ||||
| @@ -176,6 +176,7 @@ | ||||
|             <div class="row"> | ||||
|               <div class="col"> | ||||
|                 <pngx-input-check i18n-title title="Show warning when closing saved views with unsaved changes" formControlName="savedViewsWarnOnUnsavedChange"></pngx-input-check> | ||||
|                 <pngx-input-check i18n-title title="Show document counts in sidebar saved views" formControlName="sidebarViewsShowCount"></pngx-input-check> | ||||
|               </div> | ||||
|             </div> | ||||
|  | ||||
| @@ -357,6 +358,6 @@ | ||||
|  | ||||
|   <div [ngbNavOutlet]="nav" class="border-start border-end border-bottom p-3 mb-3 shadow-sm"></div> | ||||
|  | ||||
|   <button type="submit" class="btn btn-primary mb-2" [disabled]="(isDirty$ | async) === false" i18n>Save</button> | ||||
|   <button type="button" (click)="reset()" class="btn btn-secondary ms-2 mb-2" [disabled]="(isDirty$ | async) === false" i18n>Cancel</button> | ||||
|   <button type="button" (click)="reset()" class="btn btn-outline-secondary mb-2" [disabled]="(isDirty$ | async) === false" i18n>Cancel</button> | ||||
|   <button type="submit" class="btn btn-primary ms-2 mb-2" [disabled]="(isDirty$ | async) === false" i18n>Save</button> | ||||
| </form> | ||||
|   | ||||
| @@ -31,10 +31,12 @@ import { CustomDatePipe } from 'src/app/pipes/custom-date.pipe' | ||||
| import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe' | ||||
| import { PermissionsService } from 'src/app/services/permissions.service' | ||||
| import { GroupService } from 'src/app/services/rest/group.service' | ||||
| import { SavedViewService } from 'src/app/services/rest/saved-view.service' | ||||
| import { UserService } from 'src/app/services/rest/user.service' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| import { SystemStatusService } from 'src/app/services/system-status.service' | ||||
| import { Toast, ToastService } from 'src/app/services/toast.service' | ||||
| import * as navUtils from 'src/app/utils/navigation' | ||||
| import { ConfirmButtonComponent } from '../../common/confirm-button/confirm-button.component' | ||||
| import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' | ||||
| import { CheckComponent } from '../../common/input/check/check.component' | ||||
| @@ -59,6 +61,40 @@ const groups = [ | ||||
|   { id: 2, name: 'group2' }, | ||||
| ] | ||||
|  | ||||
| const status: SystemStatus = { | ||||
|   pngx_version: '2.4.3', | ||||
|   server_os: 'macOS-14.1.1-arm64-arm-64bit', | ||||
|   install_type: InstallType.BareMetal, | ||||
|   storage: { total: 494384795648, available: 13573525504 }, | ||||
|   database: { | ||||
|     type: 'sqlite', | ||||
|     url: '/paperless-ngx/data/db.sqlite3', | ||||
|     status: SystemStatusItemStatus.ERROR, | ||||
|     error: null, | ||||
|     migration_status: { | ||||
|       latest_migration: 'socialaccount.0006_alter_socialaccount_extra_data', | ||||
|       unapplied_migrations: [], | ||||
|     }, | ||||
|   }, | ||||
|   tasks: { | ||||
|     redis_url: 'redis://localhost:6379', | ||||
|     redis_status: SystemStatusItemStatus.ERROR, | ||||
|     redis_error: 'Error 61 connecting to localhost:6379. Connection refused.', | ||||
|     celery_status: SystemStatusItemStatus.ERROR, | ||||
|     celery_url: 'celery@localhost', | ||||
|     celery_error: 'Error connecting to celery@localhost', | ||||
|     index_status: SystemStatusItemStatus.OK, | ||||
|     index_last_modified: new Date().toISOString(), | ||||
|     index_error: null, | ||||
|     classifier_status: SystemStatusItemStatus.OK, | ||||
|     classifier_last_trained: new Date().toISOString(), | ||||
|     classifier_error: null, | ||||
|     sanity_check_status: SystemStatusItemStatus.ERROR, | ||||
|     sanity_check_last_run: new Date().toISOString(), | ||||
|     sanity_check_error: 'Error running sanity check.', | ||||
|   }, | ||||
| } | ||||
|  | ||||
| describe('SettingsComponent', () => { | ||||
|   let component: SettingsComponent | ||||
|   let fixture: ComponentFixture<SettingsComponent> | ||||
| @@ -72,6 +108,7 @@ describe('SettingsComponent', () => { | ||||
|   let groupService: GroupService | ||||
|   let modalService: NgbModal | ||||
|   let systemStatusService: SystemStatusService | ||||
|   let savedViewsService: SavedViewService | ||||
|  | ||||
|   beforeEach(async () => { | ||||
|     TestBed.configureTestingModule({ | ||||
| @@ -122,6 +159,7 @@ describe('SettingsComponent', () => { | ||||
|     permissionsService = TestBed.inject(PermissionsService) | ||||
|     modalService = TestBed.inject(NgbModal) | ||||
|     systemStatusService = TestBed.inject(SystemStatusService) | ||||
|     savedViewsService = TestBed.inject(SavedViewService) | ||||
|     jest.spyOn(permissionsService, 'currentUserCan').mockReturnValue(true) | ||||
|     jest | ||||
|       .spyOn(permissionsService, 'currentUserHasObjectPermissions') | ||||
| @@ -212,7 +250,7 @@ describe('SettingsComponent', () => { | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|     expect(storeSpy).toHaveBeenCalled() | ||||
|     expect(appearanceSettingsSpy).not.toHaveBeenCalled() | ||||
|     expect(setSpy).toHaveBeenCalledTimes(29) | ||||
|     expect(setSpy).toHaveBeenCalledTimes(30) | ||||
|  | ||||
|     // succeed | ||||
|     storeSpy.mockReturnValueOnce(of(true)) | ||||
| @@ -222,6 +260,9 @@ describe('SettingsComponent', () => { | ||||
|   }) | ||||
|  | ||||
|   it('should offer reload if settings changes require', () => { | ||||
|     const reloadSpy = jest | ||||
|       .spyOn(navUtils, 'locationReload') | ||||
|       .mockImplementation(() => {}) | ||||
|     completeSetup() | ||||
|     let toast: Toast | ||||
|     toastService.getToasts().subscribe((t) => (toast = t[0])) | ||||
| @@ -238,6 +279,7 @@ describe('SettingsComponent', () => { | ||||
|  | ||||
|     expect(toast.actionName).toEqual('Reload now') | ||||
|     toast.action() | ||||
|     expect(reloadSpy).toHaveBeenCalled() | ||||
|   }) | ||||
|  | ||||
|   it('should allow setting theme color, visually apply change immediately but not save', () => { | ||||
| @@ -266,7 +308,7 @@ describe('SettingsComponent', () => { | ||||
|       ) | ||||
|     completeSetup(userService) | ||||
|     fixture.detectChanges() | ||||
|     expect(toastErrorSpy).toBeCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|   }) | ||||
|  | ||||
|   it('should show errors on load if load groups failure', () => { | ||||
| @@ -278,44 +320,10 @@ describe('SettingsComponent', () => { | ||||
|       ) | ||||
|     completeSetup(groupService) | ||||
|     fixture.detectChanges() | ||||
|     expect(toastErrorSpy).toBeCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|   }) | ||||
|  | ||||
|   it('should load system status on initialize, show errors if needed', () => { | ||||
|     const status: SystemStatus = { | ||||
|       pngx_version: '2.4.3', | ||||
|       server_os: 'macOS-14.1.1-arm64-arm-64bit', | ||||
|       install_type: InstallType.BareMetal, | ||||
|       storage: { total: 494384795648, available: 13573525504 }, | ||||
|       database: { | ||||
|         type: 'sqlite', | ||||
|         url: '/paperless-ngx/data/db.sqlite3', | ||||
|         status: SystemStatusItemStatus.ERROR, | ||||
|         error: null, | ||||
|         migration_status: { | ||||
|           latest_migration: 'socialaccount.0006_alter_socialaccount_extra_data', | ||||
|           unapplied_migrations: [], | ||||
|         }, | ||||
|       }, | ||||
|       tasks: { | ||||
|         redis_url: 'redis://localhost:6379', | ||||
|         redis_status: SystemStatusItemStatus.ERROR, | ||||
|         redis_error: | ||||
|           'Error 61 connecting to localhost:6379. Connection refused.', | ||||
|         celery_status: SystemStatusItemStatus.ERROR, | ||||
|         celery_url: 'celery@localhost', | ||||
|         celery_error: 'Error connecting to celery@localhost', | ||||
|         index_status: SystemStatusItemStatus.OK, | ||||
|         index_last_modified: new Date().toISOString(), | ||||
|         index_error: null, | ||||
|         classifier_status: SystemStatusItemStatus.OK, | ||||
|         classifier_last_trained: new Date().toISOString(), | ||||
|         classifier_error: null, | ||||
|         sanity_check_status: SystemStatusItemStatus.ERROR, | ||||
|         sanity_check_last_run: new Date().toISOString(), | ||||
|         sanity_check_error: 'Error running sanity check.', | ||||
|       }, | ||||
|     } | ||||
|     jest.spyOn(systemStatusService, 'get').mockReturnValue(of(status)) | ||||
|     jest.spyOn(permissionsService, 'isAdmin').mockReturnValue(true) | ||||
|     completeSetup() | ||||
| @@ -332,6 +340,8 @@ describe('SettingsComponent', () => { | ||||
|  | ||||
|   it('should open system status dialog', () => { | ||||
|     const modalOpenSpy = jest.spyOn(modalService, 'open') | ||||
|     jest.spyOn(systemStatusService, 'get').mockReturnValue(of(status)) | ||||
|     jest.spyOn(permissionsService, 'isAdmin').mockReturnValue(true) | ||||
|     completeSetup() | ||||
|     component.showSystemStatus() | ||||
|     expect(modalOpenSpy).toHaveBeenCalledWith(SystemStatusDialogComponent, { | ||||
| @@ -345,4 +355,14 @@ describe('SettingsComponent', () => { | ||||
|     component.reset() | ||||
|     expect(component.settingsForm.get('themeColor').value).toEqual('') | ||||
|   }) | ||||
|  | ||||
|   it('should trigger maybeRefreshDocumentCounts on settings save', () => { | ||||
|     completeSetup() | ||||
|     const maybeRefreshSpy = jest.spyOn( | ||||
|       savedViewsService, | ||||
|       'maybeRefreshDocumentCounts' | ||||
|     ) | ||||
|     settingsService.settingsSaved.emit(true) | ||||
|     expect(maybeRefreshSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
|   | ||||
| @@ -2,10 +2,10 @@ import { AsyncPipe, ViewportScroller } from '@angular/common' | ||||
| import { | ||||
|   AfterViewInit, | ||||
|   Component, | ||||
|   Inject, | ||||
|   LOCALE_ID, | ||||
|   OnDestroy, | ||||
|   OnInit, | ||||
|   inject, | ||||
| } from '@angular/core' | ||||
| import { | ||||
|   FormControl, | ||||
| @@ -49,6 +49,7 @@ import { | ||||
|   PermissionsService, | ||||
| } from 'src/app/services/permissions.service' | ||||
| import { GroupService } from 'src/app/services/rest/group.service' | ||||
| import { SavedViewService } from 'src/app/services/rest/saved-view.service' | ||||
| import { UserService } from 'src/app/services/rest/user.service' | ||||
| import { | ||||
|   LanguageOption, | ||||
| @@ -56,6 +57,7 @@ import { | ||||
| } from 'src/app/services/settings.service' | ||||
| import { SystemStatusService } from 'src/app/services/system-status.service' | ||||
| import { Toast, ToastService } from 'src/app/services/toast.service' | ||||
| import { locationReload } from 'src/app/utils/navigation' | ||||
| import { CheckComponent } from '../../common/input/check/check.component' | ||||
| import { ColorComponent } from '../../common/input/color/color.component' | ||||
| import { PermissionsGroupComponent } from '../../common/input/permissions/permissions-group/permissions-group.component' | ||||
| @@ -104,6 +106,21 @@ export class SettingsComponent | ||||
|   extends ComponentWithPermissions | ||||
|   implements OnInit, AfterViewInit, OnDestroy, DirtyComponent | ||||
| { | ||||
|   private documentListViewService = inject(DocumentListViewService) | ||||
|   private toastService = inject(ToastService) | ||||
|   private settings = inject(SettingsService) | ||||
|   currentLocale = inject(LOCALE_ID) | ||||
|   private viewportScroller = inject(ViewportScroller) | ||||
|   private activatedRoute = inject(ActivatedRoute) | ||||
|   readonly tourService = inject(TourService) | ||||
|   private usersService = inject(UserService) | ||||
|   private groupsService = inject(GroupService) | ||||
|   private router = inject(Router) | ||||
|   permissionsService = inject(PermissionsService) | ||||
|   private modalService = inject(NgbModal) | ||||
|   private systemStatusService = inject(SystemStatusService) | ||||
|   private savedViewsService = inject(SavedViewService) | ||||
|  | ||||
|   activeNavID: number | ||||
|  | ||||
|   settingsForm = new FormGroup({ | ||||
| @@ -138,6 +155,7 @@ export class SettingsComponent | ||||
|     notificationsConsumerSuppressOnDashboard: new FormControl(null), | ||||
|  | ||||
|     savedViewsWarnOnUnsavedChange: new FormControl(null), | ||||
|     sidebarViewsShowCount: new FormControl(null), | ||||
|   }) | ||||
|  | ||||
|   SettingsNavIDs = SettingsNavIDs | ||||
| @@ -167,7 +185,8 @@ export class SettingsComponent | ||||
|       this.systemStatus.tasks.classifier_status === | ||||
|         SystemStatusItemStatus.ERROR || | ||||
|       this.systemStatus.tasks.sanity_check_status === | ||||
|         SystemStatusItemStatus.ERROR | ||||
|         SystemStatusItemStatus.ERROR || | ||||
|       this.systemStatus.websocket_connected === SystemStatusItemStatus.ERROR | ||||
|     ) | ||||
|   } | ||||
|  | ||||
| @@ -179,24 +198,11 @@ export class SettingsComponent | ||||
|     ) | ||||
|   } | ||||
|  | ||||
|   constructor( | ||||
|     private documentListViewService: DocumentListViewService, | ||||
|     private toastService: ToastService, | ||||
|     private settings: SettingsService, | ||||
|     @Inject(LOCALE_ID) public currentLocale: string, | ||||
|     private viewportScroller: ViewportScroller, | ||||
|     private activatedRoute: ActivatedRoute, | ||||
|     public readonly tourService: TourService, | ||||
|     private usersService: UserService, | ||||
|     private groupsService: GroupService, | ||||
|     private router: Router, | ||||
|     public permissionsService: PermissionsService, | ||||
|     private modalService: NgbModal, | ||||
|     private systemStatusService: SystemStatusService | ||||
|   ) { | ||||
|   constructor() { | ||||
|     super() | ||||
|     this.settings.settingsSaved.subscribe(() => { | ||||
|       if (!this.savePending) this.initialize() | ||||
|       this.savedViewsService.maybeRefreshDocumentCounts() | ||||
|     }) | ||||
|   } | ||||
|  | ||||
| @@ -308,6 +314,9 @@ export class SettingsComponent | ||||
|       savedViewsWarnOnUnsavedChange: this.settings.get( | ||||
|         SETTINGS_KEYS.SAVED_VIEWS_WARN_ON_UNSAVED_CHANGE | ||||
|       ), | ||||
|       sidebarViewsShowCount: this.settings.get( | ||||
|         SETTINGS_KEYS.SIDEBAR_VIEWS_SHOW_COUNT | ||||
|       ), | ||||
|       defaultPermsOwner: this.settings.get(SETTINGS_KEYS.DEFAULT_PERMS_OWNER), | ||||
|       defaultPermsViewUsers: this.settings.get( | ||||
|         SETTINGS_KEYS.DEFAULT_PERMS_VIEW_USERS | ||||
| @@ -485,6 +494,10 @@ export class SettingsComponent | ||||
|       SETTINGS_KEYS.SAVED_VIEWS_WARN_ON_UNSAVED_CHANGE, | ||||
|       this.settingsForm.value.savedViewsWarnOnUnsavedChange | ||||
|     ) | ||||
|     this.settings.set( | ||||
|       SETTINGS_KEYS.SIDEBAR_VIEWS_SHOW_COUNT, | ||||
|       this.settingsForm.value.sidebarViewsShowCount | ||||
|     ) | ||||
|     this.settings.set( | ||||
|       SETTINGS_KEYS.DEFAULT_PERMS_OWNER, | ||||
|       this.settingsForm.value.defaultPermsOwner | ||||
| @@ -539,7 +552,7 @@ export class SettingsComponent | ||||
|             savedToast.content = $localize`Settings were saved successfully. Reload is required to apply some changes.` | ||||
|             savedToast.actionName = $localize`Reload now` | ||||
|             savedToast.action = () => { | ||||
|               location.reload() | ||||
|               locationReload() | ||||
|             } | ||||
|           } | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { NgTemplateOutlet, SlicePipe } from '@angular/common' | ||||
| import { Component, OnDestroy, OnInit } from '@angular/core' | ||||
| import { Component, inject, OnDestroy, OnInit } from '@angular/core' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { Router } from '@angular/router' | ||||
| import { | ||||
| @@ -69,6 +69,10 @@ export class TasksComponent | ||||
|   extends LoadingComponentWithPermissions | ||||
|   implements OnInit, OnDestroy | ||||
| { | ||||
|   tasksService = inject(TasksService) | ||||
|   private modalService = inject(NgbModal) | ||||
|   private readonly router = inject(Router) | ||||
|  | ||||
|   public activeTab: TaskTab | ||||
|   public selectedTasks: Set<number> = new Set() | ||||
|   public togggleAll: boolean = false | ||||
| @@ -105,14 +109,6 @@ export class TasksComponent | ||||
|       : $localize`Dismiss all` | ||||
|   } | ||||
|  | ||||
|   constructor( | ||||
|     public tasksService: TasksService, | ||||
|     private modalService: NgbModal, | ||||
|     private readonly router: Router | ||||
|   ) { | ||||
|     super() | ||||
|   } | ||||
|  | ||||
|   ngOnInit() { | ||||
|     this.tasksService.reload() | ||||
|     timer(5000, 5000) | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { Component, OnDestroy } from '@angular/core' | ||||
| import { Component, OnDestroy, inject } from '@angular/core' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { Router } from '@angular/router' | ||||
| import { | ||||
| @@ -36,19 +36,19 @@ export class TrashComponent | ||||
|   extends LoadingComponentWithPermissions | ||||
|   implements OnDestroy | ||||
| { | ||||
|   private trashService = inject(TrashService) | ||||
|   private toastService = inject(ToastService) | ||||
|   private modalService = inject(NgbModal) | ||||
|   private settingsService = inject(SettingsService) | ||||
|   private router = inject(Router) | ||||
|  | ||||
|   public documentsInTrash: Document[] = [] | ||||
|   public selectedDocuments: Set<number> = new Set() | ||||
|   public allToggled: boolean = false | ||||
|   public page: number = 1 | ||||
|   public totalDocuments: number | ||||
|  | ||||
|   constructor( | ||||
|     private trashService: TrashService, | ||||
|     private toastService: ToastService, | ||||
|     private modalService: NgbModal, | ||||
|     private settingsService: SettingsService, | ||||
|     private router: Router | ||||
|   ) { | ||||
|   constructor() { | ||||
|     super() | ||||
|     this.reload() | ||||
|   } | ||||
|   | ||||
| @@ -19,6 +19,7 @@ import { GroupService } from 'src/app/services/rest/group.service' | ||||
| import { UserService } from 'src/app/services/rest/user.service' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import * as navUtils from 'src/app/utils/navigation' | ||||
| import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' | ||||
| import { GroupEditDialogComponent } from '../../common/edit-dialog/group-edit-dialog/group-edit-dialog.component' | ||||
| import { UserEditDialogComponent } from '../../common/edit-dialog/user-edit-dialog/user-edit-dialog.component' | ||||
| @@ -107,7 +108,7 @@ describe('UsersAndGroupsComponent', () => { | ||||
|     const toastErrorSpy = jest.spyOn(toastService, 'showError') | ||||
|     const toastInfoSpy = jest.spyOn(toastService, 'showInfo') | ||||
|     editDialog.failed.emit() | ||||
|     expect(toastErrorSpy).toBeCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|     settingsService.currentUser = users[1] // simulate logged in as different user | ||||
|     editDialog.succeeded.emit(users[0]) | ||||
|     expect(toastInfoSpy).toHaveBeenCalledWith( | ||||
| @@ -130,7 +131,7 @@ describe('UsersAndGroupsComponent', () => { | ||||
|       throwError(() => new Error('error deleting user')) | ||||
|     ) | ||||
|     deleteDialog.confirm() | ||||
|     expect(toastErrorSpy).toBeCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|     deleteSpy.mockReturnValueOnce(of(true)) | ||||
|     deleteDialog.confirm() | ||||
|     expect(listAllSpy).toHaveBeenCalled() | ||||
| @@ -142,19 +143,18 @@ describe('UsersAndGroupsComponent', () => { | ||||
|     let modal: NgbModalRef | ||||
|     modalService.activeInstances.subscribe((refs) => (modal = refs[0])) | ||||
|     component.editUser(users[0]) | ||||
|     const navSpy = jest | ||||
|       .spyOn(navUtils, 'setLocationHref') | ||||
|       .mockImplementation(() => {}) | ||||
|     const editDialog = modal.componentInstance as UserEditDialogComponent | ||||
|     editDialog.passwordIsSet = true | ||||
|     settingsService.currentUser = users[0] // simulate logged in as same user | ||||
|     editDialog.succeeded.emit(users[0]) | ||||
|     fixture.detectChanges() | ||||
|     Object.defineProperty(window, 'location', { | ||||
|       value: { | ||||
|         href: 'http://localhost/', | ||||
|       }, | ||||
|       writable: true, // possibility to override | ||||
|     }) | ||||
|     tick(2600) | ||||
|     expect(window.location.href).toContain('logout') | ||||
|     expect(navSpy).toHaveBeenCalledWith( | ||||
|       `${window.location.origin}/accounts/logout/?next=/accounts/login/?next=/` | ||||
|     ) | ||||
|   })) | ||||
|  | ||||
|   it('should support edit / create group, show error if needed', () => { | ||||
| @@ -166,7 +166,7 @@ describe('UsersAndGroupsComponent', () => { | ||||
|     const toastErrorSpy = jest.spyOn(toastService, 'showError') | ||||
|     const toastInfoSpy = jest.spyOn(toastService, 'showInfo') | ||||
|     editDialog.failed.emit() | ||||
|     expect(toastErrorSpy).toBeCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|     editDialog.succeeded.emit(groups[0]) | ||||
|     expect(toastInfoSpy).toHaveBeenCalledWith( | ||||
|       `Saved group "${groups[0].name}".` | ||||
| @@ -188,7 +188,7 @@ describe('UsersAndGroupsComponent', () => { | ||||
|       throwError(() => new Error('error deleting group')) | ||||
|     ) | ||||
|     deleteDialog.confirm() | ||||
|     expect(toastErrorSpy).toBeCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|     deleteSpy.mockReturnValueOnce(of(true)) | ||||
|     deleteDialog.confirm() | ||||
|     expect(listAllSpy).toHaveBeenCalled() | ||||
| @@ -210,7 +210,7 @@ describe('UsersAndGroupsComponent', () => { | ||||
|       ) | ||||
|     completeSetup(userService) | ||||
|     fixture.detectChanges() | ||||
|     expect(toastErrorSpy).toBeCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|   }) | ||||
|  | ||||
|   it('should show errors on load if load groups failure', () => { | ||||
| @@ -222,6 +222,6 @@ describe('UsersAndGroupsComponent', () => { | ||||
|       ) | ||||
|     completeSetup(groupService) | ||||
|     fixture.detectChanges() | ||||
|     expect(toastErrorSpy).toBeCalled() | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { Component, OnDestroy, OnInit } from '@angular/core' | ||||
| import { Component, OnDestroy, OnInit, inject } from '@angular/core' | ||||
| import { NgbModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { Subject, first, takeUntil } from 'rxjs' | ||||
| @@ -10,6 +10,7 @@ import { GroupService } from 'src/app/services/rest/group.service' | ||||
| import { UserService } from 'src/app/services/rest/user.service' | ||||
| import { SettingsService } from 'src/app/services/settings.service' | ||||
| import { ToastService } from 'src/app/services/toast.service' | ||||
| import { setLocationHref } from 'src/app/utils/navigation' | ||||
| import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component' | ||||
| import { EditDialogMode } from '../../common/edit-dialog/edit-dialog.component' | ||||
| import { GroupEditDialogComponent } from '../../common/edit-dialog/group-edit-dialog/group-edit-dialog.component' | ||||
| @@ -31,22 +32,18 @@ export class UsersAndGroupsComponent | ||||
|   extends ComponentWithPermissions | ||||
|   implements OnInit, OnDestroy | ||||
| { | ||||
|   private usersService = inject(UserService) | ||||
|   private groupsService = inject(GroupService) | ||||
|   private toastService = inject(ToastService) | ||||
|   private modalService = inject(NgbModal) | ||||
|   permissionsService = inject(PermissionsService) | ||||
|   private settings = inject(SettingsService) | ||||
|  | ||||
|   users: User[] | ||||
|   groups: Group[] | ||||
|  | ||||
|   unsubscribeNotifier: Subject<any> = new Subject() | ||||
|  | ||||
|   constructor( | ||||
|     private usersService: UserService, | ||||
|     private groupsService: GroupService, | ||||
|     private toastService: ToastService, | ||||
|     private modalService: NgbModal, | ||||
|     public permissionsService: PermissionsService, | ||||
|     private settings: SettingsService | ||||
|   ) { | ||||
|     super() | ||||
|   } | ||||
|  | ||||
|   ngOnInit(): void { | ||||
|     this.usersService | ||||
|       .listAll(null, null, { full_perms: true }) | ||||
| @@ -97,7 +94,9 @@ export class UsersAndGroupsComponent | ||||
|             $localize`Password has been changed, you will be logged out momentarily.` | ||||
|           ) | ||||
|           setTimeout(() => { | ||||
|             window.location.href = `${window.location.origin}/accounts/logout/?next=/accounts/login/?next=/` | ||||
|             setLocationHref( | ||||
|               `${window.location.origin}/accounts/logout/?next=/accounts/login/?next=/` | ||||
|             ) | ||||
|           }, 2500) | ||||
|         } else { | ||||
|           this.toastService.showInfo( | ||||
|   | ||||
| @@ -15,7 +15,7 @@ | ||||
|     </svg> | ||||
|     <div class="ms-2 ms-md-3 d-inline-block" [class.d-md-none]="slimSidebarEnabled"> | ||||
|       @if (customAppTitle?.length) { | ||||
|         <div class="d-flex flex-column align-items-start"> | ||||
|         <div class="d-flex flex-column align-items-start custom-title"> | ||||
|           <span class="title">{{customAppTitle}}</span> | ||||
|           <span class="byline text-uppercase font-monospace" i18n>by Paperless-ngx</span> | ||||
|         </div> | ||||
| @@ -108,11 +108,19 @@ | ||||
|                 <li class="nav-item w-100 app-link" cdkDrag [cdkDragDisabled]="!settingsService.organizingSidebarSavedViews" | ||||
|                   cdkDragPreviewContainer="parent" cdkDragPreviewClass="navItemDrag" (cdkDragStarted)="onDragStart($event)" | ||||
|                   (cdkDragEnded)="onDragEnd($event)"> | ||||
|                   <a class="nav-link" [class.text-truncate]="!slimSidebarEnabled" routerLink="view/{{view.id}}" | ||||
|                   <a class="nav-link" routerLink="view/{{view.id}}" | ||||
|                     routerLinkActive="active" (click)="closeMenu()" [ngbPopover]="view.name" | ||||
|                     [disablePopover]="!slimSidebarEnabled" placement="end" container="body" triggers="mouseenter:mouseleave" | ||||
|                     popoverClass="popover-slim"> | ||||
|                     <i-bs class="me-1" name="funnel"></i-bs><span> {{view.name}}</span> | ||||
|                     <i-bs class="me-1" name="funnel"></i-bs> | ||||
|                       <span> <div class="d-inline-flex view-name"><span class="overflow-hidden" [class.text-wrap]="!slimSidebarEnabled">{{view.name}}</span></div> | ||||
|                         @if (showSidebarCounts && !slimSidebarEnabled) { | ||||
|                           <span class="badge bg-info text-dark ms-2 d-inline">{{ savedViewService.getDocumentCount(view) }}</span> | ||||
|                         } | ||||
|                       </span> | ||||
|                     @if (showSidebarCounts && slimSidebarEnabled) { | ||||
|                       <span class="badge bg-info text-dark position-absolute top-0 end-0 d-none d-md-block">{{ savedViewService.getDocumentCount(view) }}</span> | ||||
|                     } | ||||
|                   </a> | ||||
|                   @if (settingsService.organizingSidebarSavedViews) { | ||||
|                     <div class="position-absolute end-0 top-0 px-3 py-2" [class.me-n3]="slimSidebarEnabled" cdkDragHandle> | ||||
| @@ -139,7 +147,7 @@ | ||||
|                   [disablePopover]="!slimSidebarEnabled" placement="end" container="body" triggers="mouseenter:mouseleave" | ||||
|                   popoverClass="popover-slim"> | ||||
|                   <i-bs class="me-1" name="file-text"></i-bs><span> {{d.title | documentTitle}}</span> | ||||
|                   <span class="close" (click)="closeDocument(d); $event.preventDefault()"> | ||||
|                   <span class="close flex-column justify-content-center" (click)="closeDocument(d); $event.preventDefault()"> | ||||
|                     <i-bs name="x"></i-bs> | ||||
|                   </span> | ||||
|                 </a> | ||||
|   | ||||
| @@ -19,6 +19,10 @@ | ||||
|     height: 0.8em; | ||||
|   } | ||||
|  | ||||
|   .view-name { | ||||
|     max-width: calc(100% - 50px) | ||||
|   } | ||||
|  | ||||
|   .nav-group:not(:has(.app-link)) .sidebar-heading { | ||||
|     display: none !important; | ||||
|   } | ||||
| @@ -187,7 +191,7 @@ main { | ||||
|   list-style-type: none; | ||||
|  | ||||
|   &:hover .close { | ||||
|     display: block; | ||||
|     display: flex; | ||||
|   } | ||||
|  | ||||
|   .close { | ||||
| @@ -244,7 +248,7 @@ main { | ||||
|   } | ||||
| } | ||||
|  | ||||
| @media screen and (max-width: 768px) { | ||||
| @media screen and (min-width: 366px) and (max-width: 768px) { | ||||
|   .navbar-toggler { | ||||
|     // compensate for 2 buttons on the right | ||||
|     margin-right: 45px; | ||||
| @@ -257,6 +261,13 @@ main { | ||||
|   } | ||||
| } | ||||
|  | ||||
| @media screen and (max-width: 345px) { | ||||
|   .custom-title { | ||||
|     max-width: 110px; | ||||
|     overflow: hidden; | ||||
|   } | ||||
| } | ||||
|  | ||||
| :host ::ng-deep .dropdown.show .dropdown-toggle, | ||||
| :host ::ng-deep .dropdown-toggle:hover { | ||||
|   opacity: 0.7; | ||||
|   | ||||
| @@ -92,6 +92,7 @@ describe('AppFrameComponent', () => { | ||||
|   let router: Router | ||||
|   let savedViewSpy | ||||
|   let modalService: NgbModal | ||||
|   let maybeRefreshSpy | ||||
|  | ||||
|   beforeEach(async () => { | ||||
|     TestBed.configureTestingModule({ | ||||
| @@ -113,7 +114,11 @@ describe('AppFrameComponent', () => { | ||||
|         { | ||||
|           provide: SavedViewService, | ||||
|           useValue: { | ||||
|             reload: () => {}, | ||||
|             reload: (fn: any) => { | ||||
|               if (fn) { | ||||
|                 fn() | ||||
|               } | ||||
|             }, | ||||
|             listAll: () => | ||||
|               of({ | ||||
|                 all: [saved_views.map((v) => v.id)], | ||||
| @@ -121,6 +126,8 @@ describe('AppFrameComponent', () => { | ||||
|                 results: saved_views, | ||||
|               }), | ||||
|             sidebarViews: saved_views.filter((v) => v.show_in_sidebar), | ||||
|             getDocumentCount: (view: SavedView) => 5, | ||||
|             maybeRefreshDocumentCounts: () => {}, | ||||
|           }, | ||||
|         }, | ||||
|         PermissionsService, | ||||
| @@ -169,6 +176,7 @@ describe('AppFrameComponent', () => { | ||||
|     jest.spyOn(permissionsService, 'currentUserCan').mockReturnValue(true) | ||||
|  | ||||
|     savedViewSpy = jest.spyOn(savedViewService, 'reload') | ||||
|     maybeRefreshSpy = jest.spyOn(savedViewService, 'maybeRefreshDocumentCounts') | ||||
|  | ||||
|     fixture = TestBed.createComponent(AppFrameComponent) | ||||
|     component = fixture.componentInstance | ||||
| @@ -359,4 +367,8 @@ describe('AppFrameComponent', () => { | ||||
|     expect(toastErrorSpy).toHaveBeenCalledTimes(2) | ||||
|     expect(toastInfoSpy).toHaveBeenCalledTimes(3) | ||||
|   }) | ||||
|  | ||||
|   it('should call maybeRefreshDocumentCounts after saved views reload', () => { | ||||
|     expect(maybeRefreshSpy).toHaveBeenCalled() | ||||
|   }) | ||||
| }) | ||||
|   | ||||
| @@ -6,7 +6,7 @@ import { | ||||
|   moveItemInArray, | ||||
| } from '@angular/cdk/drag-drop' | ||||
| import { NgClass } from '@angular/common' | ||||
| import { Component, HostListener, OnInit } from '@angular/core' | ||||
| import { Component, HostListener, inject, OnInit } from '@angular/core' | ||||
| import { ActivatedRoute, Router, RouterModule } from '@angular/router' | ||||
| import { | ||||
|   NgbCollapseModule, | ||||
| @@ -74,27 +74,27 @@ export class AppFrameComponent | ||||
|   extends ComponentWithPermissions | ||||
|   implements OnInit, ComponentCanDeactivate | ||||
| { | ||||
|   versionString = `${environment.appTitle} ${environment.version}` | ||||
|   router = inject(Router) | ||||
|   private activatedRoute = inject(ActivatedRoute) | ||||
|   private openDocumentsService = inject(OpenDocumentsService) | ||||
|   savedViewService = inject(SavedViewService) | ||||
|   private remoteVersionService = inject(RemoteVersionService) | ||||
|   settingsService = inject(SettingsService) | ||||
|   tasksService = inject(TasksService) | ||||
|   private readonly toastService = inject(ToastService) | ||||
|   private modalService = inject(NgbModal) | ||||
|   permissionsService = inject(PermissionsService) | ||||
|   private djangoMessagesService = inject(DjangoMessagesService) | ||||
|  | ||||
|   appRemoteVersion: AppRemoteVersion | ||||
|  | ||||
|   isMenuCollapsed: boolean = true | ||||
|  | ||||
|   slimSidebarAnimating: boolean = false | ||||
|  | ||||
|   constructor( | ||||
|     public router: Router, | ||||
|     private activatedRoute: ActivatedRoute, | ||||
|     private openDocumentsService: OpenDocumentsService, | ||||
|     public savedViewService: SavedViewService, | ||||
|     private remoteVersionService: RemoteVersionService, | ||||
|     public settingsService: SettingsService, | ||||
|     public tasksService: TasksService, | ||||
|     private readonly toastService: ToastService, | ||||
|     private modalService: NgbModal, | ||||
|     public permissionsService: PermissionsService, | ||||
|     private djangoMessagesService: DjangoMessagesService | ||||
|   ) { | ||||
|   constructor() { | ||||
|     super() | ||||
|     const permissionsService = this.permissionsService | ||||
|  | ||||
|     if ( | ||||
|       permissionsService.currentUserCan( | ||||
| @@ -102,7 +102,9 @@ export class AppFrameComponent | ||||
|         PermissionType.SavedView | ||||
|       ) | ||||
|     ) { | ||||
|       this.savedViewService.reload() | ||||
|       this.savedViewService.reload(() => { | ||||
|         this.savedViewService.maybeRefreshDocumentCounts() | ||||
|       }) | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -142,6 +144,10 @@ export class AppFrameComponent | ||||
|     }, 200) // slightly longer than css animation for slim sidebar | ||||
|   } | ||||
|  | ||||
|   get versionString(): string { | ||||
|     return `${environment.appTitle} v${this.settingsService.get(SETTINGS_KEYS.VERSION)}${environment.tag === 'prod' ? '' : ` #${environment.tag}`}` | ||||
|   } | ||||
|  | ||||
|   get customAppTitle(): string { | ||||
|     return this.settingsService.get(SETTINGS_KEYS.APP_TITLE) | ||||
|   } | ||||
| @@ -279,4 +285,11 @@ export class AppFrameComponent | ||||
|   onLogout() { | ||||
|     this.openDocumentsService.closeAll() | ||||
|   } | ||||
|  | ||||
|   get showSidebarCounts(): boolean { | ||||
|     return ( | ||||
|       this.settingsService.get(SETTINGS_KEYS.SIDEBAR_VIEWS_SHOW_COUNT) && | ||||
|       !this.settingsService.organizingSidebarSavedViews | ||||
|     ) | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -89,7 +89,7 @@ | ||||
|                 @if (searchResults?.documents.length) { | ||||
|                     <h6 class="dropdown-header" i18n="@@searchResults.documents">Documents</h6> | ||||
|                     @for (document of searchResults.documents; track document.id) { | ||||
|                         <ng-container *ngTemplateOutlet="resultItemTemplate; context: {item: document, nameProp: 'title', type: DataType.Document, icon: 'file-text', date: document.added}"></ng-container> | ||||
|                         <ng-container *ngTemplateOutlet="resultItemTemplate; context: {item: document, nameProp: 'title', type: DataType.Document, icon: 'file-text', date: document.created}"></ng-container> | ||||
|                     } | ||||
|                 } | ||||
|                 @if (searchResults?.saved_views.length) { | ||||
|   | ||||
| @@ -405,7 +405,7 @@ describe('GlobalSearchComponent', () => { | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|  | ||||
|     // succeed | ||||
|     editDialog.succeeded.emit(true) | ||||
|     editDialog.succeeded.emit(object as any) | ||||
|     expect(toastInfoSpy).toHaveBeenCalled() | ||||
|   }) | ||||
|  | ||||
| @@ -456,7 +456,7 @@ describe('GlobalSearchComponent', () => { | ||||
|     expect(toastErrorSpy).toHaveBeenCalled() | ||||
|  | ||||
|     // succeed | ||||
|     editDialog.succeeded.emit(true) | ||||
|     editDialog.succeeded.emit(searchResults.tags[0] as any) | ||||
|     expect(toastInfoSpy).toHaveBeenCalled() | ||||
|   }) | ||||
|  | ||||
| @@ -529,6 +529,17 @@ describe('GlobalSearchComponent', () => { | ||||
|     expect(dispatchSpy).toHaveBeenCalledTimes(2) // once for keydown, second for click | ||||
|   }) | ||||
|  | ||||
|   it('should support using base href in navigateOrOpenInNewWindow', () => { | ||||
|     jest | ||||
|       .spyOn(component['locationStrategy'], 'getBaseHref') | ||||
|       .mockReturnValue('/base/') | ||||
|     const openSpy = jest.spyOn(window, 'open') | ||||
|     const event = new Event('click') | ||||
|     event['ctrlKey'] = true | ||||
|     component.primaryAction(DataType.Document, { id: 1 }, event as any) | ||||
|     expect(openSpy).toHaveBeenCalledWith('/base/documents/1', '_blank') | ||||
|   }) | ||||
|  | ||||
|   it('should support title content search and advanced search', () => { | ||||
|     const qfSpy = jest.spyOn(documentListViewService, 'quickFilter') | ||||
|     component.query = 'test' | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { NgTemplateOutlet } from '@angular/common' | ||||
| import { LocationStrategy, NgTemplateOutlet } from '@angular/common' | ||||
| import { | ||||
|   Component, | ||||
|   ElementRef, | ||||
| @@ -6,6 +6,7 @@ import { | ||||
|   QueryList, | ||||
|   ViewChild, | ||||
|   ViewChildren, | ||||
|   inject, | ||||
| } from '@angular/core' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { Router } from '@angular/router' | ||||
| @@ -69,6 +70,17 @@ import { WorkflowEditDialogComponent } from '../../common/edit-dialog/workflow-e | ||||
|   ], | ||||
| }) | ||||
| export class GlobalSearchComponent implements OnInit { | ||||
|   searchService = inject(SearchService) | ||||
|   private router = inject(Router) | ||||
|   private modalService = inject(NgbModal) | ||||
|   private documentService = inject(DocumentService) | ||||
|   private documentListViewService = inject(DocumentListViewService) | ||||
|   private permissionsService = inject(PermissionsService) | ||||
|   private toastService = inject(ToastService) | ||||
|   private hotkeyService = inject(HotKeyService) | ||||
|   private settingsService = inject(SettingsService) | ||||
|   private locationStrategy = inject(LocationStrategy) | ||||
|  | ||||
|   public DataType = DataType | ||||
|   public query: string | ||||
|   public queryDebounce: Subject<string> | ||||
| @@ -90,17 +102,7 @@ export class GlobalSearchComponent implements OnInit { | ||||
|     ) | ||||
|   } | ||||
|  | ||||
|   constructor( | ||||
|     public searchService: SearchService, | ||||
|     private router: Router, | ||||
|     private modalService: NgbModal, | ||||
|     private documentService: DocumentService, | ||||
|     private documentListViewService: DocumentListViewService, | ||||
|     private permissionsService: PermissionsService, | ||||
|     private toastService: ToastService, | ||||
|     private hotkeyService: HotKeyService, | ||||
|     private settingsService: SettingsService | ||||
|   ) { | ||||
|   constructor() { | ||||
|     this.queryDebounce = new Subject<string>() | ||||
|  | ||||
|     this.queryDebounce | ||||
| @@ -421,10 +423,13 @@ export class GlobalSearchComponent implements OnInit { | ||||
|     extras: Object = {} | ||||
|   ) { | ||||
|     if (newWindow) { | ||||
|       const url = this.router.serializeUrl( | ||||
|       const serializedUrl = this.router.serializeUrl( | ||||
|         this.router.createUrlTree(commands, extras) | ||||
|       ) | ||||
|       window.open(url, '_blank') | ||||
|       const baseHref = this.locationStrategy.getBaseHref() | ||||
|       const fullUrl = | ||||
|         baseHref.replace(/\/+$/, '') + '/' + serializedUrl.replace(/^\/+/, '') | ||||
|       window.open(fullUrl, '_blank') | ||||
|     } else { | ||||
|       this.router.navigate(commands, extras) | ||||
|     } | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { Component, OnDestroy, OnInit } from '@angular/core' | ||||
| import { Component, OnDestroy, OnInit, inject } from '@angular/core' | ||||
| import { | ||||
|   NgbDropdownModule, | ||||
|   NgbProgressbarModule, | ||||
| @@ -20,7 +20,7 @@ import { ToastComponent } from '../../common/toast/toast.component' | ||||
|   ], | ||||
| }) | ||||
| export class ToastsDropdownComponent implements OnInit, OnDestroy { | ||||
|   constructor(public toastService: ToastService) {} | ||||
|   toastService = inject(ToastService) | ||||
|  | ||||
|   private subscription: Subscription | ||||
|  | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| import { DecimalPipe } from '@angular/common' | ||||
| import { Component, EventEmitter, Input, Output } from '@angular/core' | ||||
| import { Component, EventEmitter, Input, Output, inject } from '@angular/core' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { Subject } from 'rxjs' | ||||
| import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe' | ||||
| @@ -12,9 +12,7 @@ import { LoadingComponentWithPermissions } from '../../loading-component/loading | ||||
|   imports: [DecimalPipe, SafeHtmlPipe], | ||||
| }) | ||||
| export class ConfirmDialogComponent extends LoadingComponentWithPermissions { | ||||
|   constructor(public activeModal: NgbActiveModal) { | ||||
|     super() | ||||
|   } | ||||
|   activeModal = inject(NgbActiveModal) | ||||
|  | ||||
|   @Output() | ||||
|   public confirmClicked = new EventEmitter() | ||||
|   | ||||
| @@ -1,54 +0,0 @@ | ||||
| <div class="modal-header"> | ||||
|     <h4 class="modal-title" id="modal-basic-title">{{title}}</h4> | ||||
|     <button type="button" class="btn-close" aria-label="Close" (click)="cancel()"> | ||||
|     </button> | ||||
| </div> | ||||
| <div class="modal-body"> | ||||
|     <div class="row"> | ||||
|         <div class="col"> | ||||
|             <div class="btn-toolbar flex-nowrap"> | ||||
|                 <div class="input-group input-group-sm"> | ||||
|                     <div class="input-group-text" i18n>Page</div> | ||||
|                     <input class="form-control mw-60" type="number" min="1" [(ngModel)]="currentPage" /> | ||||
|                     <div class="input-group-text" i18n>of {{totalPages}}</div> | ||||
|                 </div> | ||||
|                 <div class="input-group input-group-sm ms-auto"> | ||||
|                     <span class="input-group-text" i18n>Pages to remove</span> | ||||
|                     <input [ngModel]="pagesString" class="form-control" disabled /> | ||||
|                 </div> | ||||
|             </div> | ||||
|             <div class="pdf-viewer-container w-100 mt-3"> | ||||
|                 <pdf-viewer #pdfViewer [src]="pdfSrc" [(page)]="currentPage" | ||||
|                 [original-size]="false" | ||||
|                 [zoom]="1" | ||||
|                 zoom-scale="page-fit" | ||||
|                 [render-text]="false" | ||||
|                 (pagerendered)="pageRendered($event)" | ||||
|                 (after-load-complete)="pdfPreviewLoaded($event)"> | ||||
|                 </pdf-viewer> | ||||
|             </div> | ||||
|         </div> | ||||
|     </div> | ||||
| </div> | ||||
| <div class="modal-footer flex-nowrap"> | ||||
|     <div> | ||||
|         @if (message) { | ||||
|             <p [innerHTML]="message | safeHtml"></p> | ||||
|         } | ||||
|         @if (messageBold) { | ||||
|             <p class="mb-0 small"><b [innerHTML]="messageBold | safeHtml"></b></p> | ||||
|         } | ||||
|     </div> | ||||
|     <button type="button" class="btn" [class]="cancelBtnClass" (click)="cancel()" [disabled]="!buttonsEnabled"> | ||||
|             <span class="d-inline-block" style="padding-bottom: 1px;">{{cancelBtnCaption}}</span> | ||||
|         </button> | ||||
|     <button type="button" class="btn" [class]="btnClass" (click)="confirm()" [disabled]="!confirmButtonEnabled || !buttonsEnabled"> | ||||
|         {{btnCaption}} | ||||
|     </button> | ||||
| </div> | ||||
|  | ||||
| <ng-template #pageCheckOverlay let-page="page" let-pages="pages"> | ||||
|     <div class="position-absolute top-0 start-0 w-100 h-100 p-2" (click)="pageCheckChanged(page)"> | ||||
|         <input type="checkbox" class="form-check-input" /> | ||||
|     </div> | ||||
| </ng-template> | ||||
| @@ -1,28 +0,0 @@ | ||||
| .pdf-viewer-container { | ||||
|   background-color: gray; | ||||
|   height: 550px; | ||||
|  | ||||
|   pdf-viewer { | ||||
|     width: 100%; | ||||
|     height: 100%; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .mw-60 { | ||||
|   max-width: 60px; | ||||
| } | ||||
|  | ||||
| div.position-absolute:has(.form-check-input:checked) { | ||||
|   background-color: rgba(var(--bs-dark-rgb), 0.4); | ||||
| } | ||||
|  | ||||
| .form-check-input { | ||||
|   &:checked { | ||||
|     background-color: var(--bs-danger); | ||||
|     border-color: var(--bs-danger); | ||||
|   } | ||||
|   &:focus { | ||||
|     box-shadow: 0 0 0 0.25rem rgba(var(--bs-danger-rgb), var(--pngx-focus-alpha)); | ||||
|     border-color: var(--bs-danger); | ||||
|   } | ||||
| } | ||||
| @@ -1,60 +0,0 @@ | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { provideHttpClientTesting } from '@angular/common/http/testing' | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe' | ||||
| import { DeletePagesConfirmDialogComponent } from './delete-pages-confirm-dialog.component' | ||||
|  | ||||
| describe('DeletePagesConfirmDialogComponent', () => { | ||||
|   let component: DeletePagesConfirmDialogComponent | ||||
|   let fixture: ComponentFixture<DeletePagesConfirmDialogComponent> | ||||
|  | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       declarations: [], | ||||
|       imports: [ | ||||
|         NgxBootstrapIconsModule.pick(allIcons), | ||||
|         FormsModule, | ||||
|         ReactiveFormsModule, | ||||
|         DeletePagesConfirmDialogComponent, | ||||
|       ], | ||||
|       providers: [ | ||||
|         NgbActiveModal, | ||||
|         SafeHtmlPipe, | ||||
|         provideHttpClient(withInterceptorsFromDi()), | ||||
|         provideHttpClientTesting(), | ||||
|       ], | ||||
|     }).compileComponents() | ||||
|     fixture = TestBed.createComponent(DeletePagesConfirmDialogComponent) | ||||
|     component = fixture.componentInstance | ||||
|     fixture.detectChanges() | ||||
|   }) | ||||
|  | ||||
|   it('should return a string with comma-separated pages', () => { | ||||
|     component.pages = [1, 2, 3, 4] | ||||
|     expect(component.pagesString).toEqual('1, 2, 3, 4') | ||||
|   }) | ||||
|  | ||||
|   it('should update totalPages when pdf is loaded', () => { | ||||
|     component.pdfPreviewLoaded({ numPages: 5 } as any) | ||||
|     expect(component.totalPages).toEqual(5) | ||||
|   }) | ||||
|  | ||||
|   it('should update checks when page is rendered', () => { | ||||
|     const event = { | ||||
|       target: document.createElement('div'), | ||||
|       detail: { pageNumber: 1 }, | ||||
|     } as any | ||||
|     component.pageRendered(event) | ||||
|     expect(component['checks'].length).toEqual(1) | ||||
|   }) | ||||
|  | ||||
|   it('should update pages when page check is changed', () => { | ||||
|     component.pageCheckChanged(1) | ||||
|     expect(component.pages).toEqual([1]) | ||||
|     component.pageCheckChanged(1) | ||||
|     expect(component.pages).toEqual([]) | ||||
|   }) | ||||
| }) | ||||
| @@ -1,71 +0,0 @@ | ||||
| import { Component, TemplateRef, ViewChild } from '@angular/core' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { | ||||
|   PDFDocumentProxy, | ||||
|   PdfViewerComponent, | ||||
|   PdfViewerModule, | ||||
| } from 'ng2-pdf-viewer' | ||||
| import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe' | ||||
| import { DocumentService } from 'src/app/services/rest/document.service' | ||||
| import { ConfirmDialogComponent } from '../confirm-dialog.component' | ||||
|  | ||||
| @Component({ | ||||
|   selector: 'pngx-delete-pages-confirm-dialog', | ||||
|   templateUrl: './delete-pages-confirm-dialog.component.html', | ||||
|   styleUrl: './delete-pages-confirm-dialog.component.scss', | ||||
|   imports: [PdfViewerModule, FormsModule, ReactiveFormsModule, SafeHtmlPipe], | ||||
| }) | ||||
| export class DeletePagesConfirmDialogComponent extends ConfirmDialogComponent { | ||||
|   public documentID: number | ||||
|   public pages: number[] = [] | ||||
|   public currentPage: number = 1 | ||||
|   public totalPages: number | ||||
|  | ||||
|   @ViewChild('pdfViewer') pdfViewer: PdfViewerComponent | ||||
|   @ViewChild('pageCheckOverlay') pageCheckOverlay!: TemplateRef<any> | ||||
|   private checks: HTMLElement[] = [] | ||||
|  | ||||
|   public get pagesString(): string { | ||||
|     return this.pages.join(', ') | ||||
|   } | ||||
|  | ||||
|   public get pdfSrc(): string { | ||||
|     return this.documentService.getPreviewUrl(this.documentID) | ||||
|   } | ||||
|  | ||||
|   constructor( | ||||
|     activeModal: NgbActiveModal, | ||||
|     private documentService: DocumentService | ||||
|   ) { | ||||
|     super(activeModal) | ||||
|   } | ||||
|  | ||||
|   public pdfPreviewLoaded(pdf: PDFDocumentProxy) { | ||||
|     this.totalPages = pdf.numPages | ||||
|   } | ||||
|  | ||||
|   pageRendered(event: CustomEvent) { | ||||
|     const pageDiv = event.target as HTMLDivElement | ||||
|     const check = this.pageCheckOverlay.createEmbeddedView({ | ||||
|       page: event.detail.pageNumber, | ||||
|     }) | ||||
|     this.checks[event.detail.pageNumber - 1] = check.rootNodes[0] | ||||
|     pageDiv?.insertBefore(check.rootNodes[0], pageDiv.firstChild) | ||||
|     this.updateChecks() | ||||
|   } | ||||
|  | ||||
|   pageCheckChanged(pageNumber: number) { | ||||
|     if (!this.pages.includes(pageNumber)) this.pages.push(pageNumber) | ||||
|     else if (this.pages.includes(pageNumber)) | ||||
|       this.pages.splice(this.pages.indexOf(pageNumber), 1) | ||||
|     this.updateChecks() | ||||
|   } | ||||
|  | ||||
|   private updateChecks() { | ||||
|     this.checks.forEach((check, i) => { | ||||
|       const input = check.getElementsByTagName('input')[0] | ||||
|       input.checked = this.pages.includes(i + 1) | ||||
|     }) | ||||
|   } | ||||
| } | ||||
| @@ -3,9 +3,8 @@ import { | ||||
|   DragDropModule, | ||||
|   moveItemInArray, | ||||
| } from '@angular/cdk/drag-drop' | ||||
| import { Component, OnInit } from '@angular/core' | ||||
| import { Component, OnInit, inject } from '@angular/core' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { takeUntil } from 'rxjs' | ||||
| import { Document } from 'src/app/data/document' | ||||
| @@ -28,6 +27,9 @@ export class MergeConfirmDialogComponent | ||||
|   extends ConfirmDialogComponent | ||||
|   implements OnInit | ||||
| { | ||||
|   private documentService = inject(DocumentService) | ||||
|   private permissionService = inject(PermissionsService) | ||||
|  | ||||
|   public documentIDs: number[] = [] | ||||
|   public archiveFallback: boolean = false | ||||
|   public deleteOriginals: boolean = false | ||||
| @@ -38,12 +40,8 @@ export class MergeConfirmDialogComponent | ||||
|  | ||||
|   public metadataDocumentID: number = -1 | ||||
|  | ||||
|   constructor( | ||||
|     activeModal: NgbActiveModal, | ||||
|     private documentService: DocumentService, | ||||
|     private permissionService: PermissionsService | ||||
|   ) { | ||||
|     super(activeModal) | ||||
|   constructor() { | ||||
|     super() | ||||
|   } | ||||
|  | ||||
|   ngOnInit() { | ||||
|   | ||||
| @@ -1,6 +1,5 @@ | ||||
| import { NgStyle } from '@angular/common' | ||||
| import { Component } from '@angular/core' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { Component, inject } from '@angular/core' | ||||
| import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { SafeHtmlPipe } from 'src/app/pipes/safehtml.pipe' | ||||
| import { DocumentService } from 'src/app/services/rest/document.service' | ||||
| @@ -13,6 +12,8 @@ import { ConfirmDialogComponent } from '../confirm-dialog.component' | ||||
|   imports: [NgStyle, NgxBootstrapIconsModule, SafeHtmlPipe], | ||||
| }) | ||||
| export class RotateConfirmDialogComponent extends ConfirmDialogComponent { | ||||
|   documentService = inject(DocumentService) | ||||
|  | ||||
|   public documentID: number | ||||
|   public showPDFNote: boolean = true | ||||
|  | ||||
| @@ -25,11 +26,8 @@ export class RotateConfirmDialogComponent extends ConfirmDialogComponent { | ||||
|     return degrees | ||||
|   } | ||||
|  | ||||
|   constructor( | ||||
|     activeModal: NgbActiveModal, | ||||
|     public documentService: DocumentService | ||||
|   ) { | ||||
|     super(activeModal) | ||||
|   constructor() { | ||||
|     super() | ||||
|   } | ||||
|  | ||||
|   rotate(clockwise: boolean = true) { | ||||
|   | ||||
| @@ -1,59 +0,0 @@ | ||||
| <div class="modal-header"> | ||||
|     <h4 class="modal-title" id="modal-basic-title">{{title}}</h4> | ||||
|     <button type="button" class="btn-close" aria-label="Close" (click)="cancel()"> | ||||
|     </button> | ||||
| </div> | ||||
| <div class="modal-body"> | ||||
|     <p>{{message}}</p> | ||||
|     <div class="row mb-2"> | ||||
|         <div class="col-7"> | ||||
|             <div class="input-group input-group-sm"> | ||||
|                 <div class="input-group-text" i18n>Page</div> | ||||
|                 <input class="form-control" type="number" min="1" [(ngModel)]="page" /> | ||||
|                 <div class="input-group-text" i18n>of {{totalPages}}</div> | ||||
|             </div> | ||||
|             <div class="pdf-viewer-container w-100 mt-3"> | ||||
|                 <pdf-viewer [src]="pdfSrc" [(page)]="page" | ||||
|                 [original-size]="false" | ||||
|                 [zoom]="1" | ||||
|                 zoom-scale="page-fit" | ||||
|                 (after-load-complete)="pdfPreviewLoaded($event)"> | ||||
|                 </pdf-viewer> | ||||
|             </div> | ||||
|         </div> | ||||
|         <div class="col-5"> | ||||
|             <div class="d-grid"> | ||||
|                 <button class="btn btn-sm btn-primary" (click)="addSplit()" [disabled]="!canSplit"> | ||||
|                     <i-bs name="plus-circle"></i-bs>  | ||||
|                     <span i18n>Add Split</span> | ||||
|                 </button> | ||||
|             </div> | ||||
|  | ||||
|             <ul class="list-group mt-3"> | ||||
|                 @for (pageStr of pagesString.split(','); track pageStr; let i = $index) { | ||||
|                     <li class="list-group-item d-flex align-items-center"> | ||||
|                         {{pageStr}} | ||||
|                         @if (pagesString.split(',').length > 1) { | ||||
|                               | ||||
|                             <button class="btn btn-sm btn-danger ms-auto" (click)="removeSplit(i)"> | ||||
|                                 <i-bs name="trash"></i-bs> | ||||
|                             </button> | ||||
|                         } | ||||
|                     </li> | ||||
|                 } | ||||
|             </ul> | ||||
|         </div> | ||||
|     </div> | ||||
| </div> | ||||
| <div class="modal-footer"> | ||||
|     <div class="form-check form-switch me-auto"> | ||||
|        <input class="form-check-input" type="checkbox" role="switch" id="deleteOriginalSwitch" [(ngModel)]="deleteOriginal" [disabled]="!userOwnsDocument"> | ||||
|        <label class="form-check-label" for="deleteOriginalSwitch" i18n>Delete original document after successful split</label> | ||||
|      </div> | ||||
|     <button type="button" class="btn" [class]="cancelBtnClass" (click)="cancel()" [disabled]="!buttonsEnabled"> | ||||
|             <span class="d-inline-block" style="padding-bottom: 1px;">{{cancelBtnCaption}}</span> | ||||
|         </button> | ||||
|     <button type="button" class="btn" [class]="btnClass" (click)="confirm()" [disabled]="!confirmButtonEnabled || !buttonsEnabled"> | ||||
|         {{btnCaption}} | ||||
|     </button> | ||||
| </div> | ||||
| @@ -1,9 +0,0 @@ | ||||
| .pdf-viewer-container { | ||||
|     background-color: gray; | ||||
|     height: 500px; | ||||
|  | ||||
|     pdf-viewer { | ||||
|       width: 100%; | ||||
|       height: 100%; | ||||
|     } | ||||
|   } | ||||
| @@ -1,107 +0,0 @@ | ||||
| import { ComponentFixture, TestBed } from '@angular/core/testing' | ||||
|  | ||||
| import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http' | ||||
| import { provideHttpClientTesting } from '@angular/common/http/testing' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { PdfViewerModule } from 'ng2-pdf-viewer' | ||||
| import { NgxBootstrapIconsModule, allIcons } from 'ngx-bootstrap-icons' | ||||
| import { of } from 'rxjs' | ||||
| import { DocumentService } from 'src/app/services/rest/document.service' | ||||
| import { SplitConfirmDialogComponent } from './split-confirm-dialog.component' | ||||
|  | ||||
| describe('SplitConfirmDialogComponent', () => { | ||||
|   let component: SplitConfirmDialogComponent | ||||
|   let fixture: ComponentFixture<SplitConfirmDialogComponent> | ||||
|   let documentService: DocumentService | ||||
|  | ||||
|   beforeEach(async () => { | ||||
|     await TestBed.configureTestingModule({ | ||||
|       imports: [ | ||||
|         NgxBootstrapIconsModule.pick(allIcons), | ||||
|         ReactiveFormsModule, | ||||
|         FormsModule, | ||||
|         PdfViewerModule, | ||||
|         SplitConfirmDialogComponent, | ||||
|       ], | ||||
|       providers: [ | ||||
|         NgbActiveModal, | ||||
|         provideHttpClient(withInterceptorsFromDi()), | ||||
|         provideHttpClientTesting(), | ||||
|       ], | ||||
|     }).compileComponents() | ||||
|  | ||||
|     fixture = TestBed.createComponent(SplitConfirmDialogComponent) | ||||
|     documentService = TestBed.inject(DocumentService) | ||||
|     component = fixture.componentInstance | ||||
|     fixture.detectChanges() | ||||
|   }) | ||||
|  | ||||
|   it('should load document on init', () => { | ||||
|     const getSpy = jest.spyOn(documentService, 'get') | ||||
|     component.documentID = 1 | ||||
|     getSpy.mockReturnValue(of({ id: 1 } as any)) | ||||
|     component.ngOnInit() | ||||
|     expect(documentService.get).toHaveBeenCalledWith(1) | ||||
|   }) | ||||
|  | ||||
|   it('should update pagesString when pages are added', () => { | ||||
|     component.totalPages = 5 | ||||
|     component.page = 2 | ||||
|     component.addSplit() | ||||
|     expect(component.pagesString).toEqual('1-2,3-5') | ||||
|     component.page = 4 | ||||
|     component.addSplit() | ||||
|     expect(component.pagesString).toEqual('1-2,3-4,5') | ||||
|   }) | ||||
|  | ||||
|   it('should update pagesString when pages are removed', () => { | ||||
|     component.totalPages = 5 | ||||
|     component.page = 2 | ||||
|     component.addSplit() | ||||
|     component.page = 4 | ||||
|     component.addSplit() | ||||
|     expect(component.pagesString).toEqual('1-2,3-4,5') | ||||
|     component.removeSplit(0) | ||||
|     expect(component.pagesString).toEqual('1-4,5') | ||||
|   }) | ||||
|  | ||||
|   it('should enable confirm button when pages are added', () => { | ||||
|     component.totalPages = 5 | ||||
|     component.page = 2 | ||||
|     component.addSplit() | ||||
|     expect(component.confirmButtonEnabled).toBeTruthy() | ||||
|   }) | ||||
|  | ||||
|   it('should disable confirm button when all pages are removed', () => { | ||||
|     component.totalPages = 5 | ||||
|     component.page = 2 | ||||
|     component.addSplit() | ||||
|     component.removeSplit(0) | ||||
|     expect(component.confirmButtonEnabled).toBeFalsy() | ||||
|   }) | ||||
|  | ||||
|   it('should not add split if page is the last page', () => { | ||||
|     component.totalPages = 5 | ||||
|     component.page = 5 | ||||
|     component.addSplit() | ||||
|     expect(component.pagesString).toEqual('1-5') | ||||
|   }) | ||||
|  | ||||
|   it('should update totalPages when pdf is loaded', () => { | ||||
|     component.pdfPreviewLoaded({ numPages: 5 } as any) | ||||
|     expect(component.totalPages).toEqual(5) | ||||
|   }) | ||||
|  | ||||
|   it('should correctly disable split button', () => { | ||||
|     component.totalPages = 5 | ||||
|     component.page = 1 | ||||
|     expect(component.canSplit).toBeTruthy() | ||||
|     component.page = 5 | ||||
|     expect(component.canSplit).toBeFalsy() | ||||
|     component.page = 4 | ||||
|     expect(component.canSplit).toBeTruthy() | ||||
|     component['pages'] = new Set([1, 2, 3, 4]) | ||||
|     expect(component.canSplit).toBeFalsy() | ||||
|   }) | ||||
| }) | ||||
| @@ -1,100 +0,0 @@ | ||||
| import { Component, OnInit } from '@angular/core' | ||||
| import { FormsModule, ReactiveFormsModule } from '@angular/forms' | ||||
| import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap' | ||||
| import { PDFDocumentProxy, PdfViewerModule } from 'ng2-pdf-viewer' | ||||
| import { NgxBootstrapIconsModule } from 'ngx-bootstrap-icons' | ||||
| import { Document } from 'src/app/data/document' | ||||
| import { PermissionsService } from 'src/app/services/permissions.service' | ||||
| import { DocumentService } from 'src/app/services/rest/document.service' | ||||
| import { ConfirmDialogComponent } from '../confirm-dialog.component' | ||||
|  | ||||
| @Component({ | ||||
|   selector: 'pngx-split-confirm-dialog', | ||||
|   templateUrl: './split-confirm-dialog.component.html', | ||||
|   styleUrl: './split-confirm-dialog.component.scss', | ||||
|   imports: [ | ||||
|     FormsModule, | ||||
|     ReactiveFormsModule, | ||||
|     NgxBootstrapIconsModule, | ||||
|     PdfViewerModule, | ||||
|   ], | ||||
| }) | ||||
| export class SplitConfirmDialogComponent | ||||
|   extends ConfirmDialogComponent | ||||
|   implements OnInit | ||||
| { | ||||
|   public get pagesString(): string { | ||||
|     let pagesStr = '' | ||||
|  | ||||
|     let lastPage = 1 | ||||
|     for (let i = 1; i <= this.totalPages; i++) { | ||||
|       if (this.pages.has(i) || i === this.totalPages) { | ||||
|         if (lastPage === i) { | ||||
|           pagesStr += `${i},` | ||||
|           lastPage = Math.min(i + 1, this.totalPages) | ||||
|         } else { | ||||
|           pagesStr += `${lastPage}-${i},` | ||||
|           lastPage = Math.min(i + 1, this.totalPages) | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     return pagesStr.replace(/,$/, '') | ||||
|   } | ||||
|  | ||||
|   private pages: Set<number> = new Set() | ||||
|  | ||||
|   public documentID: number | ||||
|   private document: Document | ||||
|   public page: number = 1 | ||||
|   public totalPages: number | ||||
|   public deleteOriginal: boolean = false | ||||
|  | ||||
|   public get canSplit(): boolean { | ||||
|     return ( | ||||
|       this.page < this.totalPages && | ||||
|       this.pages.size < this.totalPages - 1 && | ||||
|       !this.pages.has(this.page) | ||||
|     ) | ||||
|   } | ||||
|  | ||||
|   public get pdfSrc(): string { | ||||
|     return this.documentService.getPreviewUrl(this.documentID) | ||||
|   } | ||||
|  | ||||
|   constructor( | ||||
|     activeModal: NgbActiveModal, | ||||
|     private documentService: DocumentService, | ||||
|     private permissionService: PermissionsService | ||||
|   ) { | ||||
|     super(activeModal) | ||||
|     this.confirmButtonEnabled = this.pages.size > 0 | ||||
|   } | ||||
|  | ||||
|   ngOnInit(): void { | ||||
|     this.documentService.get(this.documentID).subscribe((r) => { | ||||
|       this.document = r | ||||
|     }) | ||||
|   } | ||||
|  | ||||
|   pdfPreviewLoaded(pdf: PDFDocumentProxy) { | ||||
|     this.totalPages = pdf.numPages | ||||
|   } | ||||
|  | ||||
|   addSplit() { | ||||
|     if (this.page === this.totalPages) return | ||||
|     this.pages.add(this.page) | ||||
|     this.pages = new Set(Array.from(this.pages).sort((a, b) => a - b)) | ||||
|     this.confirmButtonEnabled = this.pages.size > 0 | ||||
|   } | ||||
|  | ||||
|   removeSplit(i: number) { | ||||
|     let page = Array.from(this.pages)[Math.min(i, this.pages.size - 1)] | ||||
|     this.pages.delete(page) | ||||
|     this.confirmButtonEnabled = this.pages.size > 0 | ||||
|   } | ||||
|  | ||||
|   get userOwnsDocument(): boolean { | ||||
|     return this.permissionService.currentUserOwnsObject(this.document) | ||||
|   } | ||||
| } | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user