mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-22 12:11:19 +00:00 
			
		
		
		
	Compare commits
	
		
			731 Commits
		
	
	
		
			develop-20
			...
			develop-20
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 7f827fb277 | ||
|  | 8e700944fd | ||
|  | 037a128942 | ||
|  | ca364dc877 | ||
|  | e55af7186c | ||
|  | 354bbebbee | ||
|  | a70fab1e87 | ||
|  | b9c93091cd | ||
|  | 4349f8f303 | ||
|  | 5088e20f25 | ||
|  | 3f81aa7403 | ||
|  | 8885d1dbeb | ||
|  | a6ba75d528 | ||
|  | 667cbb1332 | ||
|  | d1bae875f7 | ||
|  | c5cf529413 | ||
|  | 62b9f2785f | ||
|  | 3b85f87502 | ||
|  | 87d3d14504 | ||
|  | 5637573fd0 | ||
|  | 48255ae6ee | ||
|  | ea02986170 | ||
|  | f4fbc15ac6 | ||
|  | a02c4b42a4 | ||
|  | 1ff47441ce | ||
|  | 2cc8568077 | ||
|  | 408687ec6b | ||
|  | 308abffb0b | ||
|  | 6743b3fe83 | ||
|  | ac0113e445 | ||
|  | 0f0a28c3d9 | ||
|  | 79f2d70211 | ||
|  | 8a06c0f7ec | ||
|  | c54da62005 | ||
|  | e5923202af | ||
|  | eb6f78406e | ||
|  | d61f87f649 | ||
|  | 33dcce7525 | ||
|  | b1d86c3a37 | ||
|  | 822dee6e70 | ||
|  | f6dac83777 | ||
|  | d3c557ca22 | ||
|  | 853a99852e | ||
|  | 8f24ac4fcd | ||
|  | 8b09cfb8c9 | ||
|  | 18ae950d2e | ||
|  | 69dfbda847 | ||
|  | 4ec2fcdb8a | ||
|  | 08879d31ba | ||
|  | 66d09450d3 | ||
|  | 74f7c07a76 | ||
|  | ae3d0a3e49 | ||
|  | 61e8d7d7a2 | ||
|  | 62c5440605 | ||
|  | 0aa90b9453 | ||
|  | 855bc2f8e7 | ||
|  | d8f05492c3 | ||
|  | 4a264f34fa | ||
|  | 5a1413e758 | ||
|  | 84dbeeb0ce | ||
|  | d868dc0945 | ||
|  | beecf9c229 | ||
|  | e39ba46398 | ||
|  | e6b6a3cee5 | ||
|  | b5483f6ad3 | ||
|  | a751218d53 | ||
|  | 2af5e6eeef | ||
|  | 013c43f9f2 | ||
|  | 7e08a1f33c | ||
|  | e592b56d7a | ||
|  | a2479f71fe | ||
|  | 7d3b993b98 | ||
|  | 90623101a3 | ||
|  | e2eca79b25 | ||
|  | 8c0ee8f024 | ||
|  | 69cae3ae55 | ||
|  | 8a06298385 | ||
|  | acc3c294d8 | ||
|  | dbf7dba421 | ||
|  | 65813f290d | ||
|  | 3491fbb99d | ||
|  | cb6b3d5f85 | ||
|  | 956d4e09c3 | ||
|  | 6a7c35e7bc | ||
|  | 090aecb5f5 | ||
|  | b653d63d3d | ||
|  | 258dbf4a98 | ||
|  | 53335077ff | ||
|  | ecfb3e2f95 | ||
|  | f512e6724e | ||
|  | de9efb0727 | ||
|  | 9075fa8ac8 | ||
|  | 768bd892c8 | ||
|  | 9d9483e20f | ||
|  | 935453796e | ||
|  | c2d3f5da16 | ||
|  | 9e6f9d16e4 | ||
|  | fad016f92f | ||
|  | 30df6684cb | ||
|  | 4aa911420a | ||
|  | 19555a7046 | ||
|  | e5c409a8fc | ||
|  | 3adf3d2fdb | ||
|  | 0923d5a23e | ||
|  | 76b8cdc385 | ||
|  | 0a27da83eb | ||
|  | 17d6e2be85 | ||
|  | 7381f3eba9 | ||
|  | 7f2ef1b8e1 | ||
|  | 9dccae2402 | ||
|  | 073afd5b6e | ||
|  | 4167d85be2 | ||
|  | ee28d1307d | ||
|  | 7562215666 | ||
|  | 0203b918e9 | ||
|  | ae7c664418 | ||
|  | f13e0991fb | ||
|  | deae94b658 | ||
|  | c38c752520 | ||
|  | 28e7df2527 | ||
|  | cb0b42e44b | ||
|  | a3674c4dfe | ||
|  | 27480561ee | ||
|  | 6ea7152423 | ||
|  | dab95f7a86 | ||
|  | adf34805a8 | ||
|  | 93e926465f | ||
|  | 5b0be91f93 | ||
|  | 01e7b604da | ||
|  | 974a550d22 | ||
|  | 58d175444b | ||
|  | 29d8861e96 | ||
|  | eb832c750f | ||
|  | 134770644a | ||
|  | a9f21c9371 | ||
|  | 781947beeb | ||
|  | 9760cd2f97 | ||
|  | d317e9ec32 | ||
|  | 534f7fcadb | ||
|  | fb3f7a1d4b | ||
|  | bf2c3e3561 | ||
|  | b670f81dcd | ||
|  | 7aac1cdf67 | ||
|  | fa0ac8a16c | ||
|  | 0990b1f0b4 | ||
|  | c1922670c8 | ||
|  | 81cd89d66f | ||
|  | f5c202543c | ||
|  | 034f437c6b | ||
|  | 8550ba6138 | ||
|  | a51501025b | ||
|  | a9d26e4586 | ||
|  | 949691935f | ||
|  | 4835b05304 | ||
|  | cce5a73dd2 | ||
|  | 3152f635dd | ||
|  | 12e8651017 | ||
|  | f88286c848 | ||
|  | 39cf0533d9 | ||
|  | 6d82887960 | ||
|  | 262f1bae34 | ||
|  | 75dfdcc220 | ||
|  | c3a3bdf525 | ||
|  | 602df95f3c | ||
|  | 296a64e284 | ||
|  | b87b99a755 | ||
|  | bf53f5d6b7 | ||
|  | bcbb868acc | ||
|  | b3a9e6569e | ||
|  | f174f124ef | ||
|  | 8745377f31 | ||
|  | cf76e93f31 | ||
|  | 19e36f6f6e | ||
|  | 2a7bb6f048 | ||
|  | 536eacbc0c | ||
|  | af78158d0b | ||
|  | fb97910a34 | ||
|  | 66b4d14129 | ||
|  | fd53047dbc | ||
|  | 5f6fc1bab4 | ||
|  | cd0b64bd24 | ||
|  | 91b26bce0e | ||
|  | 26a8bd921d | ||
|  | ad77d55c16 | ||
|  | 445a383dd0 | ||
|  | 1666af939e | ||
|  | 35447f5eee | ||
|  | 7c6fcb5731 | ||
|  | fabd9bf765 | ||
|  | 6352d26633 | ||
|  | ebef145bd6 | ||
|  | acc89eb5f9 | ||
|  | 6523596415 | ||
|  | b6c2d23116 | ||
|  | 2a123354f9 | ||
|  | 1e7ea4b76c | ||
|  | d959526eb3 | ||
|  | 8846ee9091 | ||
|  | 6eb8d0fc8c | ||
|  | 1b0e16b6a5 | ||
|  | 2e4df28288 | ||
|  | f3b7a3015d | ||
|  | 5de5e08b1d | ||
|  | 0a116cd04c | ||
|  | fd32a692c1 | ||
|  | 1ac762aba8 | ||
|  | 315dc532b6 | ||
|  | e19ed1be15 | ||
|  | cbb0621fd9 | ||
|  | 049cbab861 | ||
|  | 28b620fb5c | ||
|  | c183f91ff6 | ||
|  | 172efae41c | ||
|  | 756e857ba0 | ||
|  | 1cde7aab0c | ||
|  | 2d67eece5d | ||
|  | b1f79c4c0f | ||
|  | 33bd2ceae8 | ||
|  | e68850f192 | ||
|  | 450ac7e6ee | ||
|  | 91f52b5dbc | ||
|  | eed2405d76 | ||
|  | c956df7790 | ||
|  | 0fdccec6a8 | ||
|  | 8ded54d7a8 | ||
|  | bb1b4ca5ca | ||
|  | e90d60113b | ||
|  | d95dada0e0 | ||
|  | 8722456595 | ||
|  | b5ad226451 | ||
|  | ddb0e66651 | ||
|  | e802899608 | ||
|  | 0894d3bf42 | ||
|  | 80bcfd3bcd | ||
|  | 8c410f42bd | ||
|  | 7a1021dffc | ||
|  | 63883c9a84 | ||
|  | 50d7f9d1ec | ||
|  | ebc7ea0eb6 | ||
|  | b905efd0aa | ||
|  | 93085599b7 | ||
|  | 8a8bbaf827 | ||
|  | 96a66b894a | ||
|  | 1ba641c279 | ||
|  | 6c5ddfcb8a | ||
|  | 65ddc246dc | ||
|  | e4aff5ff4c | ||
|  | dfece91541 | ||
|  | 6ddda13c3a | ||
|  | 46219c4678 | ||
|  | a1595d0647 | ||
|  | 4ee9f9bb27 | ||
|  | bcaa0bddea | ||
|  | 01cce49070 | ||
|  | 293be04d40 | ||
|  | 44a00ec8eb | ||
|  | d84e772945 | ||
|  | b475f6c51d | ||
|  | 2712662510 | ||
|  | 1f2eeba862 | ||
|  | 46a60134f4 | ||
|  | f3f7820816 | ||
|  | a57f8076b2 | ||
|  | 517afa2273 | ||
|  | 2142b23aec | ||
|  | 1bec43b111 | ||
|  | d2978a5ee8 | ||
|  | 39b61c71e8 | ||
|  | fa2c964790 | ||
|  | 134aeb3a46 | ||
|  | 6f6e1a4ff4 | ||
|  | b743bf3d9e | ||
|  | 84ee6f16c9 | ||
|  | 9fe39e42b3 | ||
|  | 4013c7e316 | ||
|  | 0b76747531 | ||
|  | 3129756b37 | ||
|  | b0df383004 | ||
|  | 9c5b1df86c | ||
|  | 5854e24775 | ||
|  | 3f873422f2 | ||
|  | f898990773 | ||
|  | a5759ab1c6 | ||
|  | 9bd6417e02 | ||
|  | cd12a10214 | ||
|  | ee8cb62e04 | ||
|  | 87ee95a852 | ||
|  | 10f8436885 | ||
|  | 6955846a1c | ||
|  | 11d2f8d471 | ||
|  | 99347ffff1 | ||
|  | 3ddc11a905 | ||
|  | 0a48c0c20f | ||
|  | 8bc764d6ef | ||
|  | 4b4f568558 | ||
|  | d42117281a | ||
|  | d68ed5a713 | ||
|  | 2f6f36c3f0 | ||
|  | 984aa02e35 | ||
|  | cdadc7d533 | ||
|  | 4d59955cc5 | ||
|  | fc547ba59a | ||
|  | 1b2ded3167 | ||
|  | dcf20a472a | ||
|  | 7bd528defe | ||
|  | 9e6d123165 | ||
|  | c592f51c83 | ||
|  | fbb6f30366 | ||
|  | 4546c721fb | ||
|  | 3a659c9a81 | ||
|  | e28a988eb3 | ||
|  | a29742fe1f | ||
|  | dd06838eec | ||
|  | 5eb828bff8 | ||
|  | c7f3701053 | ||
|  | ab6799442c | ||
|  | 1fe9bf7d76 | ||
|  | bd14797da6 | ||
|  | 93b4e6a8d0 | ||
|  | 5566671971 | ||
|  | defcc7a00c | ||
|  | 5183de634b | ||
|  | 7771b0311c | ||
|  | 52abe3bbc2 | ||
|  | 34db9f41c2 | ||
|  | 72c31fbe6a | ||
|  | 1d8dd41564 | ||
|  | 3409240a19 | ||
|  | eccc58e75a | ||
|  | 24098f35bb | ||
|  | e7d9dc57d8 | ||
|  | f04ed5b8f0 | ||
|  | 8bd44f429b | ||
|  | fb7866b165 | ||
|  | 98db6db1eb | ||
|  | 0a235ec523 | ||
|  | cb4f7ad0a8 | ||
|  | 91ce1d3b88 | ||
|  | 8f9ab150ee | ||
|  | 8d52f83b4f | ||
|  | fb19739b54 | ||
|  | acd4f28ae3 | ||
|  | e4d443db50 | ||
|  | fe9d36cd27 | ||
|  | a916cc7e78 | ||
|  | 77d15f884b | ||
|  | ab773c9052 | ||
|  | 87d292ca27 | ||
|  | 946e272d1c | ||
|  | ffe2e310aa | ||
|  | 9fc6af0f2d | ||
|  | 71807b67fc | ||
|  | ab61d0ce21 | ||
|  | f2dc0d234b | ||
|  | 4885dbc78e | ||
|  | 0e29e282df | ||
|  | 0b3fd335ad | ||
|  | 9b2263c7bb | ||
|  | 1305fafd38 | ||
|  | 844b8d48c4 | ||
|  | fc9ef290f1 | ||
|  | 020c8ad933 | ||
|  | 33ea27b411 | ||
|  | b39e3e5965 | ||
|  | ba7e504f12 | ||
|  | a732ce2a20 | ||
|  | 2b76c4801f | ||
|  | dd09dd59e9 | ||
|  | d56088d086 | ||
|  | f27f00cfa0 | ||
|  | d762136e0b | ||
|  | 536cd78e29 | ||
|  | 5ef0889f50 | ||
|  | c2ffedb506 | ||
|  | fe702a3030 | ||
|  | 2244d366bd | ||
|  | ac16e0294e | ||
|  | bf1b0ee3c2 | ||
|  | ffec4bbfba | ||
|  | ae6e103c09 | ||
|  | 99d4471f75 | ||
|  | 7f1fb72298 | ||
|  | 009c2ed5d9 | ||
|  | 44fc00299c | ||
|  | 774fc4281c | ||
|  | c47955c069 | ||
|  | e7569644f7 | ||
|  | c567474043 | ||
|  | 9f394e92fe | ||
|  | 66befc7e44 | ||
|  | c3a28fc698 | ||
|  | ef317d5b3c | ||
|  | 152301f9ee | ||
|  | 645e9ba1f7 | ||
|  | 56487c3a33 | ||
|  | b8062a915c | ||
|  | 5780c9512a | ||
|  | 71d39707d9 | ||
|  | 9ccb8ae692 | ||
|  | 8cd50bb5bd | ||
|  | ae9e1278e5 | ||
|  | 58c03797b2 | ||
|  | 7db38b4c6c | ||
|  | da6b447e64 | ||
|  | c19ac2b0f3 | ||
|  | d5ca2171b3 | ||
|  | 20972cb29f | ||
|  | 7b714d0866 | ||
|  | 240ae8fa57 | ||
|  | 5a2f6b2652 | ||
|  | 4196ce31f0 | ||
|  | be8ca5db50 | ||
|  | 30a417ea3c | ||
|  | 695ed940e0 | ||
|  | 1353554cf8 | ||
|  | e1ba2732af | ||
|  | 42b57c0e0e | ||
|  | a6072753b2 | ||
|  | e92c224c39 | ||
|  | a3ed7ec8f6 | ||
|  | 17a2f99dff | ||
|  | c14971543c | ||
|  | 55f899608d | ||
|  | 83be63f27e | ||
|  | ed48d190e5 | ||
|  | 3c3b6615e6 | ||
|  | e71e5a877b | ||
|  | b2a65dc660 | ||
|  | d66dccd076 | ||
|  | c1128b28f2 | ||
|  | da8e78c28d | ||
|  | f50aa6b0ce | ||
|  | 661e4e53e6 | ||
|  | 3eeda4a6aa | ||
|  | 4dba9cea21 | ||
|  | 6aab5fab05 | ||
|  | 4b0597d19a | ||
|  | 92f534bcb3 | ||
|  | 76e91be4dc | ||
|  | deca4fed56 | ||
|  | 73512b0365 | ||
|  | aaffc125e7 | ||
|  | 41a48c39a0 | ||
|  | 2d96bd84b5 | ||
|  | ad1c1d2254 | ||
|  | 813206766d | ||
|  | bb25d4a82a | ||
|  | f3b78beecc | ||
|  | 64073768fe | ||
|  | fe6dd0f901 | ||
|  | aac8d11ff6 | ||
|  | afa99a35b5 | ||
|  | e9cb0a51d7 | ||
|  | 9fbcccfd02 | ||
|  | 468c9c9d56 | ||
|  | f76b27a73d | ||
|  | 579fe81616 | ||
|  | ec9ba53690 | ||
|  | 85337c53d4 | ||
|  | eb6d585bb2 | ||
|  | 378ffbc609 | ||
|  | 3b3c8e5bcd | ||
|  | 75cbdb6a57 | ||
|  | cdaff0d983 | ||
|  | dda3863889 | ||
|  | 57dc423b3f | ||
|  | c0570bc3b2 | ||
|  | 65110d1666 | ||
|  | 5a10b29402 | ||
|  | 028544ca2e | ||
|  | af46729372 | ||
|  | 4c7789a668 | ||
|  | 292908048c | ||
|  | e300314e05 | ||
|  | 4f1f360346 | ||
|  | bff856aeff | ||
|  | 7f5a1bda8d | ||
|  | b506281bd6 | ||
|  | dfe9b3e787 | ||
|  | 2428a2a7c5 | ||
|  | 0e8f608e00 | ||
|  | 70071767ab | ||
|  | 0ad6beb66c | ||
|  | 1197f65589 | ||
|  | 7bbf2dcc6f | ||
|  | d4ab69ebe6 | ||
|  | c8c552602e | ||
|  | 1921a8050b | ||
|  | f488feda93 | ||
|  | d90c033b83 | ||
|  | 9f256253f2 | ||
|  | 489b7c12e5 | ||
|  | 1049a8314d | ||
|  | 48301b6b9c | ||
|  | 5a92215921 | ||
|  | ccfc75852a | ||
|  | 9804cffff3 | ||
|  | 901e113fef | ||
|  | a4021ff056 | ||
|  | 902d91ad29 | ||
|  | fa2cf22e73 | ||
|  | 970dad4c49 | ||
|  | 9d01c7bdb8 | ||
|  | dc7d4fb258 | ||
|  | a807ca5002 | ||
|  | d59d326841 | ||
|  | b915548e82 | ||
|  | 8200a81840 | ||
|  | 6a49918707 | ||
|  | e55fc483bd | ||
|  | 4ff5f5883d | ||
|  | eda2eae04a | ||
|  | c07c30ea17 | ||
|  | 56f1eb03e0 | ||
|  | d4e14dd262 | ||
|  | 0c7f04fb17 | ||
|  | 061c01da53 | ||
|  | 716d72d8af | ||
|  | 3233ca4a4c | ||
|  | 1041030b1e | ||
|  | bb3b06cf08 | ||
|  | f35e361915 | ||
|  | 47d697c7dc | ||
|  | 3745d79f1f | ||
|  | 04cbff4b9a | ||
|  | 0c2ca4b97c | ||
|  | 3918665cd1 | ||
|  | 9eb8869649 | ||
|  | 62221af591 | ||
|  | 4d013a44ce | ||
|  | c55cfd1acf | ||
|  | b2652b83ce | ||
|  | 3d28932216 | ||
|  | 87567d5a31 | ||
|  | 37d45f4f87 | ||
|  | 4acf0828e4 | ||
|  | b5bab53e7a | ||
|  | 60b0dc6279 | ||
|  | 883994de19 | ||
|  | 62d72516ba | ||
|  | cfb86c683e | ||
|  | 6278662014 | ||
|  | 65dcad6898 | ||
|  | 8e7bcbdd7b | ||
|  | 084ce02c21 | ||
|  | ea6addafe6 | ||
|  | da36d84b79 | ||
|  | d6d9f665c7 | ||
|  | 73d2621255 | ||
|  | 4299ba033c | ||
|  | 6007687bcc | ||
|  | 6d3f2bd5e7 | ||
|  | 210bd83bd4 | ||
|  | 65a926874b | ||
|  | d0e7681e56 | ||
|  | 33bde854ec | ||
|  | da874c97f4 | ||
|  | aa7ec95a59 | ||
|  | 4dd77303f7 | ||
|  | 166d4bd842 | ||
|  | 54bf56bbb0 | ||
|  | 3d8c6671f4 | ||
|  | 4e33de9c29 | ||
|  | 27d46eb3fa | ||
|  | e1e7a2e497 | ||
|  | a476b4259c | ||
|  | 0c03ec5ddd | ||
|  | 2d95b7f8ef | ||
|  | 34cc79ce4f | ||
|  | 6ff9943fe0 | ||
|  | a0166b45e4 | ||
|  | d27b035b20 | ||
|  | f3b387fc22 | ||
|  | 03904ffcde | ||
|  | a7973190c2 | ||
|  | 671ff95f22 | ||
|  | 01181ceea9 | ||
|  | 97643639d1 | ||
|  | 424783c47b | ||
|  | ea0ced70b2 | ||
|  | 1a633e64ef | ||
|  | 30da3f4399 | ||
|  | 6bdff95d87 | ||
|  | 895ae279d5 | ||
|  | 8b57f45963 | ||
|  | a2d2b7edd3 | ||
|  | 3d65f00c6e | ||
|  | 333004c4d9 | ||
|  | 7451659824 | ||
|  | ec89c23ace | ||
|  | 92d07d346f | ||
|  | a7ac894af2 | ||
|  | 20f89e3a7c | ||
|  | 5d2f11c3c7 | ||
|  | 1eea79e431 | ||
|  | 1aaaac67ca | ||
|  | 1eb86639c9 | ||
|  | 28b911c4ee | ||
|  | c525c70ec0 | ||
|  | 1f7d6e218b | ||
|  | a69b6d9ce2 | ||
|  | 28b2ddde18 | ||
|  | a16cc73c77 | ||
|  | 46395e350a | ||
|  | f62e49090c | ||
|  | af3b40a314 | ||
|  | c3cea0fa9e | ||
|  | 6cbdb2ce70 | ||
|  | b78460100d | ||
|  | bf6e1cb0e1 | ||
|  | 6a53f5031c | ||
|  | ae15ec01e8 | ||
|  | fe3c7c47c4 | ||
|  | 68b934010c | ||
|  | 22852bd238 | ||
|  | 5b3b1804f3 | ||
|  | f2588eb343 | ||
|  | 64a643ceec | ||
|  | 1add505644 | ||
|  | 9663eb6a19 | ||
|  | f30a24a02f | ||
|  | 68655d60a6 | ||
|  | 63b0efcd81 | ||
|  | 93284682c8 | ||
|  | 3bafcb6ad2 | ||
|  | 942d027556 | ||
|  | a60882d5f5 | ||
|  | 680f554981 | ||
|  | 20e4dc07ce | ||
|  | 184d8eb027 | ||
|  | 59725b088a | ||
|  | 32fca4a9f5 | ||
|  | 7dccf6ec48 | ||
|  | 917665feac | ||
|  | 06c50b68c2 | ||
|  | 7035c399d8 | ||
|  | 7c0ac5805c | ||
|  | 3424741583 | ||
|  | baf0297994 | ||
|  | 31d06752fa | ||
|  | 8a27154798 | ||
|  | 6d87e38ec0 | ||
|  | ccdc30a6fb | ||
|  | 90005538d3 | ||
|  | f4e0428ebc | ||
|  | bd1326eca9 | ||
|  | bdfa834251 | ||
|  | 4a9aeb4e44 | ||
|  | 3886c0fbde | ||
|  | d998eff56e | ||
|  | d73df9bf0a | ||
|  | 754f2f3a34 | ||
|  | 43fd7c928a | ||
|  | 05768c2e73 | ||
|  | 3feb2c9955 | ||
|  | 7d9f3ac473 | ||
|  | 8a5755c8f1 | ||
|  | a75a760019 | ||
|  | 78d1a130d2 | ||
|  | 79fd43f32b | ||
|  | 7a7bd65a27 | ||
|  | 01a9ecccac | ||
|  | c1b3c71090 | ||
|  | 323d04fe6c | ||
|  | 80c2f1ea3f | ||
|  | 784494871d | ||
|  | 8e93af5cc7 | ||
|  | ebfdeeedaa | ||
|  | 1d02ed6a56 | ||
|  | cca53cb0e8 | ||
|  | ab33aee4b1 | ||
|  | 41d4fab071 | ||
|  | f0a1913dc6 | ||
|  | b956b463c2 | ||
|  | 43603c4990 | ||
|  | 196e738f60 | ||
|  | 59e2ea357a | ||
|  | 5e9d942069 | ||
|  | 53d5bedd85 | ||
|  | 49c68af07b | ||
|  | c84c8e1aef | ||
|  | a8d43d7174 | ||
|  | 1087278890 | ||
|  | ae5912ab52 | ||
|  | 035bd96ae5 | ||
|  | 7283c616a0 | ||
|  | 5706666bb6 | ||
|  | 4607466fb6 | ||
|  | 34bcfcfe9b | ||
|  | 05986cb6a6 | ||
|  | 0c4ee9f043 | ||
|  | ff222795cf | ||
|  | e0c76695ee | ||
|  | ae126e8322 | ||
|  | 8f9c35fbe8 | ||
|  | 84efd6e2ee | ||
|  | b1fbe4e909 | ||
|  | 8576877072 | ||
|  | c298aced01 | ||
|  | ac61a78d8d | ||
|  | fce90a94c4 | ||
|  | 023a4f178d | ||
|  | ef254995ad | ||
|  | faeb74634b | ||
|  | b5baae373f | ||
|  | 63de711cda | ||
|  | bd28d116cb | ||
|  | 7efc2861bc | ||
|  | 5c689a2ca2 | ||
|  | d5d65df76f | ||
|  | df7d4f700c | ||
|  | efebe3cb41 | ||
|  | 2ba5b6ae49 | ||
|  | 31d93efab2 | ||
|  | 657b95485c | ||
|  | 3bfc12f93b | ||
|  | ccfd2f2ac3 | ||
|  | fb3fe0d87b | ||
|  | c80c6d52fe | ||
|  | 0fb3c0c7bf | ||
|  | 43c625bee2 | ||
|  | 967a5dd256 | ||
|  | 3c9d1bcaa1 | ||
|  | 8cd2de14a9 | ||
|  | 7728a35e04 | ||
|  | 49000da123 | ||
|  | 32a8f89875 | ||
|  | 49e5a81bd3 | ||
|  | a8efe86af0 | ||
|  | 4f07b089d2 | ||
|  | e786bf47c2 | 
| @@ -19,26 +19,25 @@ | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| 
 | ||||
| use PhpCsFixer\Runner\Parallel\ParallelConfigFactory; | ||||
| 
 | ||||
| $current = __DIR__; | ||||
| 
 | ||||
| $paths = [ | ||||
|     $current . '/../../app', | ||||
|     $current . '/../../config', | ||||
|     $current . '/../../database', | ||||
|     $current . '/../../routes', | ||||
|     $current . '/../../tests', | ||||
|     $current . '/../../resources/lang/en_US', | ||||
| ]; | ||||
| 
 | ||||
| $finder = PhpCsFixer\Finder::create() | ||||
|                            ->in($paths); | ||||
| 
 | ||||
| 
 | ||||
| $config = new PhpCsFixer\Config(); | ||||
| $config->setParallelConfig(ParallelConfigFactory::detect()); | ||||
| $config = (new PhpCsFixer\Config()) | ||||
|         // ->setUnsupportedPhpVersionAllowed(true) // use this when PHP 8.5 comes out.
 | ||||
|         ->setParallelConfig(PhpCsFixer\Runner\Parallel\ParallelConfigFactory::detect()) | ||||
|         ; | ||||
| return $config->setRules( | ||||
| 
 | ||||
|     [ | ||||
|         // rule sets
 | ||||
|         '@PHP83Migration'               => true, | ||||
| @@ -53,9 +52,6 @@ return $config->setRules( | ||||
|         'statement_indentation'         => true, | ||||
|         'void_return'                   => true, | ||||
| 
 | ||||
|         // about importing statements
 | ||||
|         'global_namespace_import' => ['import_classes' => true, 'import_constants' => true, 'import_functions' => true], | ||||
| 
 | ||||
|         // disabled rules
 | ||||
|         'native_function_invocation'    => false, // annoying
 | ||||
|         'php_unit_data_provider_name'   => false, // bloody annoying long test names
 | ||||
| @@ -64,9 +60,15 @@ return $config->setRules( | ||||
|         'comment_to_phpdoc'             => false, // breaks phpstan lines in combination with PHPStorm.
 | ||||
|         'type_declaration_spaces'       => false, | ||||
|         'cast_spaces'                   => false, | ||||
|         'phpdoc_to_comment'             => false, // do not overrule single line comment style, breaks phpstan.
 | ||||
| 
 | ||||
|         // enabled rules
 | ||||
|         'global_namespace_import' => true, // matches with rector.
 | ||||
| 
 | ||||
|         // complex rules
 | ||||
|         'phpdoc_to_comment' => ['ignored_tags' => ['var']], | ||||
|         'php_unit_test_case_static_method_calls' => [ | ||||
|             'call_type' => 'this', | ||||
|         ], | ||||
|         'array_syntax'                  => ['syntax' => 'short'], | ||||
|         'binary_operator_spaces'        => [ | ||||
|             'default'   => 'at_least_single_space', | ||||
| @@ -76,5 +78,7 @@ return $config->setRules( | ||||
|                 '??=' => 'align_single_space_minimal_by_scope', | ||||
|             ], | ||||
|         ], | ||||
|     ]) | ||||
|     ] | ||||
| 
 | ||||
| ) | ||||
|               ->setFinder($finder); | ||||
|   | ||||
							
								
								
									
										371
									
								
								.ci/php-cs-fixer/composer.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										371
									
								
								.ci/php-cs-fixer/composer.lock
									
									
									
										generated
									
									
									
								
							| @@ -151,16 +151,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "composer/semver", | ||||
|             "version": "3.4.3", | ||||
|             "version": "3.4.4", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/composer/semver.git", | ||||
|                 "reference": "4313d26ada5e0c4edfbd1dc481a92ff7bff91f12" | ||||
|                 "reference": "198166618906cb2de69b95d7d47e5fa8aa1b2b95" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/composer/semver/zipball/4313d26ada5e0c4edfbd1dc481a92ff7bff91f12", | ||||
|                 "reference": "4313d26ada5e0c4edfbd1dc481a92ff7bff91f12", | ||||
|                 "url": "https://api.github.com/repos/composer/semver/zipball/198166618906cb2de69b95d7d47e5fa8aa1b2b95", | ||||
|                 "reference": "198166618906cb2de69b95d7d47e5fa8aa1b2b95", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -212,7 +212,7 @@ | ||||
|             "support": { | ||||
|                 "irc": "ircs://irc.libera.chat:6697/composer", | ||||
|                 "issues": "https://github.com/composer/semver/issues", | ||||
|                 "source": "https://github.com/composer/semver/tree/3.4.3" | ||||
|                 "source": "https://github.com/composer/semver/tree/3.4.4" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -222,13 +222,9 @@ | ||||
|                 { | ||||
|                     "url": "https://github.com/composer", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/composer/composer", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-09-19T14:15:21+00:00" | ||||
|             "time": "2025-08-20T19:15:30+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "composer/xdebug-handler", | ||||
| @@ -345,16 +341,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "fidry/cpu-core-counter", | ||||
|             "version": "1.2.0", | ||||
|             "version": "1.3.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/theofidry/cpu-core-counter.git", | ||||
|                 "reference": "8520451a140d3f46ac33042715115e290cf5785f" | ||||
|                 "reference": "db9508f7b1474469d9d3c53b86f817e344732678" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/theofidry/cpu-core-counter/zipball/8520451a140d3f46ac33042715115e290cf5785f", | ||||
|                 "reference": "8520451a140d3f46ac33042715115e290cf5785f", | ||||
|                 "url": "https://api.github.com/repos/theofidry/cpu-core-counter/zipball/db9508f7b1474469d9d3c53b86f817e344732678", | ||||
|                 "reference": "db9508f7b1474469d9d3c53b86f817e344732678", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -364,10 +360,10 @@ | ||||
|                 "fidry/makefile": "^0.2.0", | ||||
|                 "fidry/php-cs-fixer-config": "^1.1.2", | ||||
|                 "phpstan/extension-installer": "^1.2.0", | ||||
|                 "phpstan/phpstan": "^1.9.2", | ||||
|                 "phpstan/phpstan-deprecation-rules": "^1.0.0", | ||||
|                 "phpstan/phpstan-phpunit": "^1.2.2", | ||||
|                 "phpstan/phpstan-strict-rules": "^1.4.4", | ||||
|                 "phpstan/phpstan": "^2.0", | ||||
|                 "phpstan/phpstan-deprecation-rules": "^2.0.0", | ||||
|                 "phpstan/phpstan-phpunit": "^2.0", | ||||
|                 "phpstan/phpstan-strict-rules": "^2.0", | ||||
|                 "phpunit/phpunit": "^8.5.31 || ^9.5.26", | ||||
|                 "webmozarts/strict-phpunit": "^7.5" | ||||
|             }, | ||||
| @@ -394,7 +390,7 @@ | ||||
|             ], | ||||
|             "support": { | ||||
|                 "issues": "https://github.com/theofidry/cpu-core-counter/issues", | ||||
|                 "source": "https://github.com/theofidry/cpu-core-counter/tree/1.2.0" | ||||
|                 "source": "https://github.com/theofidry/cpu-core-counter/tree/1.3.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -402,62 +398,62 @@ | ||||
|                     "type": "github" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-08-06T10:04:20+00:00" | ||||
|             "time": "2025-08-14T07:29:31+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "friendsofphp/php-cs-fixer", | ||||
|             "version": "v3.75.0", | ||||
|             "version": "v3.88.2", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer.git", | ||||
|                 "reference": "399a128ff2fdaf4281e4e79b755693286cdf325c" | ||||
|                 "reference": "a8d15584bafb0f0d9d938827840060fd4a3ebc99" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/PHP-CS-Fixer/PHP-CS-Fixer/zipball/399a128ff2fdaf4281e4e79b755693286cdf325c", | ||||
|                 "reference": "399a128ff2fdaf4281e4e79b755693286cdf325c", | ||||
|                 "url": "https://api.github.com/repos/PHP-CS-Fixer/PHP-CS-Fixer/zipball/a8d15584bafb0f0d9d938827840060fd4a3ebc99", | ||||
|                 "reference": "a8d15584bafb0f0d9d938827840060fd4a3ebc99", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "clue/ndjson-react": "^1.0", | ||||
|                 "clue/ndjson-react": "^1.3", | ||||
|                 "composer/semver": "^3.4", | ||||
|                 "composer/xdebug-handler": "^3.0.3", | ||||
|                 "composer/xdebug-handler": "^3.0.5", | ||||
|                 "ext-filter": "*", | ||||
|                 "ext-hash": "*", | ||||
|                 "ext-json": "*", | ||||
|                 "ext-tokenizer": "*", | ||||
|                 "fidry/cpu-core-counter": "^1.2", | ||||
|                 "fidry/cpu-core-counter": "^1.3", | ||||
|                 "php": "^7.4 || ^8.0", | ||||
|                 "react/child-process": "^0.6.5", | ||||
|                 "react/event-loop": "^1.0", | ||||
|                 "react/promise": "^2.0 || ^3.0", | ||||
|                 "react/socket": "^1.0", | ||||
|                 "react/stream": "^1.0", | ||||
|                 "sebastian/diff": "^4.0 || ^5.1 || ^6.0 || ^7.0", | ||||
|                 "symfony/console": "^5.4 || ^6.4 || ^7.0", | ||||
|                 "symfony/event-dispatcher": "^5.4 || ^6.4 || ^7.0", | ||||
|                 "symfony/filesystem": "^5.4 || ^6.4 || ^7.0", | ||||
|                 "symfony/finder": "^5.4 || ^6.4 || ^7.0", | ||||
|                 "symfony/options-resolver": "^5.4 || ^6.4 || ^7.0", | ||||
|                 "symfony/polyfill-mbstring": "^1.31", | ||||
|                 "symfony/polyfill-php80": "^1.31", | ||||
|                 "symfony/polyfill-php81": "^1.31", | ||||
|                 "symfony/process": "^5.4 || ^6.4 || ^7.2", | ||||
|                 "symfony/stopwatch": "^5.4 || ^6.4 || ^7.0" | ||||
|                 "react/child-process": "^0.6.6", | ||||
|                 "react/event-loop": "^1.5", | ||||
|                 "react/promise": "^3.3", | ||||
|                 "react/socket": "^1.16", | ||||
|                 "react/stream": "^1.4", | ||||
|                 "sebastian/diff": "^4.0.6 || ^5.1.1 || ^6.0.2 || ^7.0", | ||||
|                 "symfony/console": "^5.4.47 || ^6.4.24 || ^7.0", | ||||
|                 "symfony/event-dispatcher": "^5.4.45 || ^6.4.24 || ^7.0", | ||||
|                 "symfony/filesystem": "^5.4.45 || ^6.4.24 || ^7.0", | ||||
|                 "symfony/finder": "^5.4.45 || ^6.4.24 || ^7.0", | ||||
|                 "symfony/options-resolver": "^5.4.45 || ^6.4.24 || ^7.0", | ||||
|                 "symfony/polyfill-mbstring": "^1.33", | ||||
|                 "symfony/polyfill-php80": "^1.33", | ||||
|                 "symfony/polyfill-php81": "^1.33", | ||||
|                 "symfony/polyfill-php84": "^1.33", | ||||
|                 "symfony/process": "^5.4.47 || ^6.4.24 || ^7.2", | ||||
|                 "symfony/stopwatch": "^5.4.45 || ^6.4.24 || ^7.0" | ||||
|             }, | ||||
|             "require-dev": { | ||||
|                 "facile-it/paraunit": "^1.3.1 || ^2.6", | ||||
|                 "infection/infection": "^0.29.14", | ||||
|                 "justinrainbow/json-schema": "^5.3 || ^6.2", | ||||
|                 "keradus/cli-executor": "^2.1", | ||||
|                 "facile-it/paraunit": "^1.3.1 || ^2.7", | ||||
|                 "infection/infection": "^0.31.0", | ||||
|                 "justinrainbow/json-schema": "^6.5", | ||||
|                 "keradus/cli-executor": "^2.2", | ||||
|                 "mikey179/vfsstream": "^1.6.12", | ||||
|                 "php-coveralls/php-coveralls": "^2.7", | ||||
|                 "php-cs-fixer/accessible-object": "^1.1", | ||||
|                 "php-coveralls/php-coveralls": "^2.8", | ||||
|                 "php-cs-fixer/phpunit-constraint-isidenticalstring": "^1.6", | ||||
|                 "php-cs-fixer/phpunit-constraint-xmlmatchesxsd": "^1.6", | ||||
|                 "phpunit/phpunit": "^9.6.22 || ^10.5.45 || ^11.5.12", | ||||
|                 "symfony/var-dumper": "^5.4.48 || ^6.4.18 || ^7.2.3", | ||||
|                 "symfony/yaml": "^5.4.45 || ^6.4.18 || ^7.2.3" | ||||
|                 "phpunit/phpunit": "^9.6.25 || ^10.5.53 || ^11.5.34", | ||||
|                 "symfony/var-dumper": "^5.4.48 || ^6.4.24 || ^7.3.2", | ||||
|                 "symfony/yaml": "^5.4.45 || ^6.4.24 || ^7.3.2" | ||||
|             }, | ||||
|             "suggest": { | ||||
|                 "ext-dom": "For handling output formats in XML", | ||||
| @@ -498,7 +494,7 @@ | ||||
|             ], | ||||
|             "support": { | ||||
|                 "issues": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/issues", | ||||
|                 "source": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/tree/v3.75.0" | ||||
|                 "source": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/tree/v3.88.2" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -506,7 +502,7 @@ | ||||
|                     "type": "github" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-03-31T18:40:42+00:00" | ||||
|             "time": "2025-09-27T00:24:15+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "psr/container", | ||||
| @@ -958,23 +954,23 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "react/promise", | ||||
|             "version": "v3.2.0", | ||||
|             "version": "v3.3.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/reactphp/promise.git", | ||||
|                 "reference": "8a164643313c71354582dc850b42b33fa12a4b63" | ||||
|                 "reference": "23444f53a813a3296c1368bb104793ce8d88f04a" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/reactphp/promise/zipball/8a164643313c71354582dc850b42b33fa12a4b63", | ||||
|                 "reference": "8a164643313c71354582dc850b42b33fa12a4b63", | ||||
|                 "url": "https://api.github.com/repos/reactphp/promise/zipball/23444f53a813a3296c1368bb104793ce8d88f04a", | ||||
|                 "reference": "23444f53a813a3296c1368bb104793ce8d88f04a", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "php": ">=7.1.0" | ||||
|             }, | ||||
|             "require-dev": { | ||||
|                 "phpstan/phpstan": "1.10.39 || 1.4.10", | ||||
|                 "phpstan/phpstan": "1.12.28 || 1.4.10", | ||||
|                 "phpunit/phpunit": "^9.6 || ^7.5" | ||||
|             }, | ||||
|             "type": "library", | ||||
| @@ -1019,7 +1015,7 @@ | ||||
|             ], | ||||
|             "support": { | ||||
|                 "issues": "https://github.com/reactphp/promise/issues", | ||||
|                 "source": "https://github.com/reactphp/promise/tree/v3.2.0" | ||||
|                 "source": "https://github.com/reactphp/promise/tree/v3.3.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1027,7 +1023,7 @@ | ||||
|                     "type": "open_collective" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-05-24T10:39:05+00:00" | ||||
|             "time": "2025-08-19T18:57:03+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "react/socket", | ||||
| @@ -1256,16 +1252,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/console", | ||||
|             "version": "v7.3.0", | ||||
|             "version": "v7.3.4", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/console.git", | ||||
|                 "reference": "66c1440edf6f339fd82ed6c7caa76cb006211b44" | ||||
|                 "reference": "2b9c5fafbac0399a20a2e82429e2bd735dcfb7db" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/console/zipball/66c1440edf6f339fd82ed6c7caa76cb006211b44", | ||||
|                 "reference": "66c1440edf6f339fd82ed6c7caa76cb006211b44", | ||||
|                 "url": "https://api.github.com/repos/symfony/console/zipball/2b9c5fafbac0399a20a2e82429e2bd735dcfb7db", | ||||
|                 "reference": "2b9c5fafbac0399a20a2e82429e2bd735dcfb7db", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -1330,7 +1326,7 @@ | ||||
|                 "terminal" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/console/tree/v7.3.0" | ||||
|                 "source": "https://github.com/symfony/console/tree/v7.3.4" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1341,12 +1337,16 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-05-24T10:34:04+00:00" | ||||
|             "time": "2025-09-22T15:31:00+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/deprecation-contracts", | ||||
| @@ -1417,16 +1417,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/event-dispatcher", | ||||
|             "version": "v7.3.0", | ||||
|             "version": "v7.3.3", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/event-dispatcher.git", | ||||
|                 "reference": "497f73ac996a598c92409b44ac43b6690c4f666d" | ||||
|                 "reference": "b7dc69e71de420ac04bc9ab830cf3ffebba48191" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/497f73ac996a598c92409b44ac43b6690c4f666d", | ||||
|                 "reference": "497f73ac996a598c92409b44ac43b6690c4f666d", | ||||
|                 "url": "https://api.github.com/repos/symfony/event-dispatcher/zipball/b7dc69e71de420ac04bc9ab830cf3ffebba48191", | ||||
|                 "reference": "b7dc69e71de420ac04bc9ab830cf3ffebba48191", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -1477,7 +1477,7 @@ | ||||
|             "description": "Provides tools that allow your application components to communicate with each other by dispatching events and listening to them", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/event-dispatcher/tree/v7.3.0" | ||||
|                 "source": "https://github.com/symfony/event-dispatcher/tree/v7.3.3" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1488,12 +1488,16 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-04-22T09:11:45+00:00" | ||||
|             "time": "2025-08-13T11:49:31+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/event-dispatcher-contracts", | ||||
| @@ -1573,16 +1577,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/filesystem", | ||||
|             "version": "v7.3.0", | ||||
|             "version": "v7.3.2", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/filesystem.git", | ||||
|                 "reference": "b8dce482de9d7c9fe2891155035a7248ab5c7fdb" | ||||
|                 "reference": "edcbb768a186b5c3f25d0643159a787d3e63b7fd" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/filesystem/zipball/b8dce482de9d7c9fe2891155035a7248ab5c7fdb", | ||||
|                 "reference": "b8dce482de9d7c9fe2891155035a7248ab5c7fdb", | ||||
|                 "url": "https://api.github.com/repos/symfony/filesystem/zipball/edcbb768a186b5c3f25d0643159a787d3e63b7fd", | ||||
|                 "reference": "edcbb768a186b5c3f25d0643159a787d3e63b7fd", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -1619,7 +1623,7 @@ | ||||
|             "description": "Provides basic utilities for the filesystem", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/filesystem/tree/v7.3.0" | ||||
|                 "source": "https://github.com/symfony/filesystem/tree/v7.3.2" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1630,25 +1634,29 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-10-25T15:15:23+00:00" | ||||
|             "time": "2025-07-07T08:17:47+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/finder", | ||||
|             "version": "v7.3.0", | ||||
|             "version": "v7.3.2", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/finder.git", | ||||
|                 "reference": "ec2344cf77a48253bbca6939aa3d2477773ea63d" | ||||
|                 "reference": "2a6614966ba1074fa93dae0bc804227422df4dfe" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/finder/zipball/ec2344cf77a48253bbca6939aa3d2477773ea63d", | ||||
|                 "reference": "ec2344cf77a48253bbca6939aa3d2477773ea63d", | ||||
|                 "url": "https://api.github.com/repos/symfony/finder/zipball/2a6614966ba1074fa93dae0bc804227422df4dfe", | ||||
|                 "reference": "2a6614966ba1074fa93dae0bc804227422df4dfe", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -1683,7 +1691,7 @@ | ||||
|             "description": "Finds files and directories via an intuitive fluent interface", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/finder/tree/v7.3.0" | ||||
|                 "source": "https://github.com/symfony/finder/tree/v7.3.2" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1694,25 +1702,29 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-12-30T19:00:26+00:00" | ||||
|             "time": "2025-07-15T13:41:35+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/options-resolver", | ||||
|             "version": "v7.3.0", | ||||
|             "version": "v7.3.3", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/options-resolver.git", | ||||
|                 "reference": "afb9a8038025e5dbc657378bfab9198d75f10fca" | ||||
|                 "reference": "0ff2f5c3df08a395232bbc3c2eb7e84912df911d" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/options-resolver/zipball/afb9a8038025e5dbc657378bfab9198d75f10fca", | ||||
|                 "reference": "afb9a8038025e5dbc657378bfab9198d75f10fca", | ||||
|                 "url": "https://api.github.com/repos/symfony/options-resolver/zipball/0ff2f5c3df08a395232bbc3c2eb7e84912df911d", | ||||
|                 "reference": "0ff2f5c3df08a395232bbc3c2eb7e84912df911d", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -1750,7 +1762,7 @@ | ||||
|                 "options" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/options-resolver/tree/v7.3.0" | ||||
|                 "source": "https://github.com/symfony/options-resolver/tree/v7.3.3" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1761,16 +1773,20 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-04-04T13:12:05+00:00" | ||||
|             "time": "2025-08-05T10:16:07+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/polyfill-ctype", | ||||
|             "version": "v1.32.0", | ||||
|             "version": "v1.33.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/polyfill-ctype.git", | ||||
| @@ -1829,7 +1845,7 @@ | ||||
|                 "portable" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/polyfill-ctype/tree/v1.32.0" | ||||
|                 "source": "https://github.com/symfony/polyfill-ctype/tree/v1.33.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1840,6 +1856,10 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
| @@ -1849,16 +1869,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/polyfill-intl-grapheme", | ||||
|             "version": "v1.32.0", | ||||
|             "version": "v1.33.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/polyfill-intl-grapheme.git", | ||||
|                 "reference": "b9123926e3b7bc2f98c02ad54f6a4b02b91a8abe" | ||||
|                 "reference": "380872130d3a5dd3ace2f4010d95125fde5d5c70" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/polyfill-intl-grapheme/zipball/b9123926e3b7bc2f98c02ad54f6a4b02b91a8abe", | ||||
|                 "reference": "b9123926e3b7bc2f98c02ad54f6a4b02b91a8abe", | ||||
|                 "url": "https://api.github.com/repos/symfony/polyfill-intl-grapheme/zipball/380872130d3a5dd3ace2f4010d95125fde5d5c70", | ||||
|                 "reference": "380872130d3a5dd3ace2f4010d95125fde5d5c70", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -1907,7 +1927,7 @@ | ||||
|                 "shim" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/polyfill-intl-grapheme/tree/v1.32.0" | ||||
|                 "source": "https://github.com/symfony/polyfill-intl-grapheme/tree/v1.33.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1918,16 +1938,20 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2024-09-09T11:45:10+00:00" | ||||
|             "time": "2025-06-27T09:58:17+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/polyfill-intl-normalizer", | ||||
|             "version": "v1.32.0", | ||||
|             "version": "v1.33.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/polyfill-intl-normalizer.git", | ||||
| @@ -1988,7 +2012,7 @@ | ||||
|                 "shim" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/polyfill-intl-normalizer/tree/v1.32.0" | ||||
|                 "source": "https://github.com/symfony/polyfill-intl-normalizer/tree/v1.33.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -1999,6 +2023,10 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
| @@ -2008,7 +2036,7 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/polyfill-mbstring", | ||||
|             "version": "v1.32.0", | ||||
|             "version": "v1.33.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/polyfill-mbstring.git", | ||||
| @@ -2069,7 +2097,7 @@ | ||||
|                 "shim" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/polyfill-mbstring/tree/v1.32.0" | ||||
|                 "source": "https://github.com/symfony/polyfill-mbstring/tree/v1.33.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -2080,6 +2108,10 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
| @@ -2089,7 +2121,7 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/polyfill-php80", | ||||
|             "version": "v1.32.0", | ||||
|             "version": "v1.33.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/polyfill-php80.git", | ||||
| @@ -2149,7 +2181,7 @@ | ||||
|                 "shim" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/polyfill-php80/tree/v1.32.0" | ||||
|                 "source": "https://github.com/symfony/polyfill-php80/tree/v1.33.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -2160,6 +2192,10 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
| @@ -2169,7 +2205,7 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/polyfill-php81", | ||||
|             "version": "v1.32.0", | ||||
|             "version": "v1.33.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/polyfill-php81.git", | ||||
| @@ -2225,7 +2261,7 @@ | ||||
|                 "shim" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/polyfill-php81/tree/v1.32.0" | ||||
|                 "source": "https://github.com/symfony/polyfill-php81/tree/v1.33.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -2236,6 +2272,10 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
| @@ -2244,17 +2284,97 @@ | ||||
|             "time": "2024-09-09T11:45:10+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/process", | ||||
|             "version": "v7.3.0", | ||||
|             "name": "symfony/polyfill-php84", | ||||
|             "version": "v1.33.0", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/process.git", | ||||
|                 "reference": "40c295f2deb408d5e9d2d32b8ba1dd61e36f05af" | ||||
|                 "url": "https://github.com/symfony/polyfill-php84.git", | ||||
|                 "reference": "d8ced4d875142b6a7426000426b8abc631d6b191" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/process/zipball/40c295f2deb408d5e9d2d32b8ba1dd61e36f05af", | ||||
|                 "reference": "40c295f2deb408d5e9d2d32b8ba1dd61e36f05af", | ||||
|                 "url": "https://api.github.com/repos/symfony/polyfill-php84/zipball/d8ced4d875142b6a7426000426b8abc631d6b191", | ||||
|                 "reference": "d8ced4d875142b6a7426000426b8abc631d6b191", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
|                 "php": ">=7.2" | ||||
|             }, | ||||
|             "type": "library", | ||||
|             "extra": { | ||||
|                 "thanks": { | ||||
|                     "url": "https://github.com/symfony/polyfill", | ||||
|                     "name": "symfony/polyfill" | ||||
|                 } | ||||
|             }, | ||||
|             "autoload": { | ||||
|                 "files": [ | ||||
|                     "bootstrap.php" | ||||
|                 ], | ||||
|                 "psr-4": { | ||||
|                     "Symfony\\Polyfill\\Php84\\": "" | ||||
|                 }, | ||||
|                 "classmap": [ | ||||
|                     "Resources/stubs" | ||||
|                 ] | ||||
|             }, | ||||
|             "notification-url": "https://packagist.org/downloads/", | ||||
|             "license": [ | ||||
|                 "MIT" | ||||
|             ], | ||||
|             "authors": [ | ||||
|                 { | ||||
|                     "name": "Nicolas Grekas", | ||||
|                     "email": "p@tchwork.com" | ||||
|                 }, | ||||
|                 { | ||||
|                     "name": "Symfony Community", | ||||
|                     "homepage": "https://symfony.com/contributors" | ||||
|                 } | ||||
|             ], | ||||
|             "description": "Symfony polyfill backporting some PHP 8.4+ features to lower PHP versions", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "keywords": [ | ||||
|                 "compatibility", | ||||
|                 "polyfill", | ||||
|                 "portable", | ||||
|                 "shim" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/polyfill-php84/tree/v1.33.0" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
|                     "url": "https://symfony.com/sponsor", | ||||
|                     "type": "custom" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-06-24T13:30:11+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/process", | ||||
|             "version": "v7.3.4", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/process.git", | ||||
|                 "reference": "f24f8f316367b30810810d4eb30c543d7003ff3b" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/process/zipball/f24f8f316367b30810810d4eb30c543d7003ff3b", | ||||
|                 "reference": "f24f8f316367b30810810d4eb30c543d7003ff3b", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -2286,7 +2406,7 @@ | ||||
|             "description": "Executes commands in sub-processes", | ||||
|             "homepage": "https://symfony.com", | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/process/tree/v7.3.0" | ||||
|                 "source": "https://github.com/symfony/process/tree/v7.3.4" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -2297,12 +2417,16 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-04-17T09:11:12+00:00" | ||||
|             "time": "2025-09-11T10:12:26+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/service-contracts", | ||||
| @@ -2451,16 +2575,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "symfony/string", | ||||
|             "version": "v7.3.0", | ||||
|             "version": "v7.3.4", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/symfony/string.git", | ||||
|                 "reference": "f3570b8c61ca887a9e2938e85cb6458515d2b125" | ||||
|                 "reference": "f96476035142921000338bad71e5247fbc138872" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/symfony/string/zipball/f3570b8c61ca887a9e2938e85cb6458515d2b125", | ||||
|                 "reference": "f3570b8c61ca887a9e2938e85cb6458515d2b125", | ||||
|                 "url": "https://api.github.com/repos/symfony/string/zipball/f96476035142921000338bad71e5247fbc138872", | ||||
|                 "reference": "f96476035142921000338bad71e5247fbc138872", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -2475,7 +2599,6 @@ | ||||
|             }, | ||||
|             "require-dev": { | ||||
|                 "symfony/emoji": "^7.1", | ||||
|                 "symfony/error-handler": "^6.4|^7.0", | ||||
|                 "symfony/http-client": "^6.4|^7.0", | ||||
|                 "symfony/intl": "^6.4|^7.0", | ||||
|                 "symfony/translation-contracts": "^2.5|^3.0", | ||||
| @@ -2518,7 +2641,7 @@ | ||||
|                 "utf8" | ||||
|             ], | ||||
|             "support": { | ||||
|                 "source": "https://github.com/symfony/string/tree/v7.3.0" | ||||
|                 "source": "https://github.com/symfony/string/tree/v7.3.4" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -2529,12 +2652,16 @@ | ||||
|                     "url": "https://github.com/fabpot", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://github.com/nicolas-grekas", | ||||
|                     "type": "github" | ||||
|                 }, | ||||
|                 { | ||||
|                     "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", | ||||
|                     "type": "tidelift" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-04-20T20:19:01+00:00" | ||||
|             "time": "2025-09-11T14:36:48+00:00" | ||||
|         } | ||||
|     ], | ||||
|     "packages-dev": [], | ||||
|   | ||||
| @@ -26,9 +26,10 @@ SCRIPT_DIR="$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) | ||||
| cd $SCRIPT_DIR/php-cs-fixer | ||||
| composer update --quiet | ||||
| rm -f .php-cs-fixer.cache | ||||
| PHP_CS_FIXER_IGNORE_ENV=true ./vendor/bin/php-cs-fixer fix \ | ||||
| ./vendor/bin/php-cs-fixer fix \ | ||||
|     --config $SCRIPT_DIR/php-cs-fixer/.php-cs-fixer.php \ | ||||
|     --format=txt -v \ | ||||
|     --format=txt \ | ||||
|     -v \ | ||||
|     --allow-risky=yes | ||||
|  | ||||
| EXIT_CODE=$? | ||||
|   | ||||
| @@ -1,6 +1,4 @@ | ||||
| parameters: | ||||
|   scanFiles: | ||||
|     - ../_ide_helper | ||||
|   paths: | ||||
|     - ../app | ||||
|     - ../database | ||||
| @@ -9,28 +7,24 @@ parameters: | ||||
|     - ../bootstrap/app.php | ||||
|   universalObjectCratesClasses: | ||||
|     - Illuminate\Database\Eloquent\Model | ||||
|   # TODO: slowly remove these parameters and fix the issues found. | ||||
|   reportUnmatchedIgnoredErrors: true | ||||
|   ignoreErrors: | ||||
|   # TODO: slowly remove these exceptions and fix the issues found. | ||||
|     - '#Dynamic call to static method#' # all the Laravel ORM things depend on this. | ||||
|     - identifier: varTag.nativeType | ||||
|     - identifier: varTag.type | ||||
|     # all errors below I will never fix. | ||||
|     - '#expects view-string\|null, string given#' | ||||
|     - '#expects view-string, string given#' | ||||
|     - "#Parameter \\#[1-2] \\$num[1-2] of function bc[a-z]+ expects numeric-string, [a-z\\-|&]+ given#" | ||||
|     - identifier: missingType.generics # not interesting enough to fix. | ||||
|     - | ||||
|         identifier: larastan.noEnvCallsOutsideOfConfig | ||||
|         path: ../app/Console/Commands/System/CreatesDatabase.php | ||||
|     - identifier: missingType.iterableValue # not interesting enough to fix. | ||||
|     - identifier: missingType.generics # not interesting enough to fix. | ||||
|     - "#Parameter \\#[1-2] \\$num[1-2] of function bc[a-z]+ expects numeric-string, [a-z\\-|&]+ given#" | ||||
|     - '#expects view-string, string given#' | ||||
|     - '#expects view-string\|null, string given#' | ||||
|  | ||||
|     # phpstan can't handle this so we ignore them. | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::before#' | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::after#' | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::withTrashed#' | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::accountTypeIn#' | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\BelongsTo::withTrashed#' | ||||
|     - identifier: varTag.type # needs a custom extension for every repository, not gonna happen. | ||||
|     - '#Dynamic call to static method Illuminate#' | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::before#' # is custom scope | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::after#' # is custom scope | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::withTrashed#' # is to allow soft delete | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::accountTypeIn#' # is a custom scope | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\BelongsTo::withTrashed#' # is to allow soft delete | ||||
|  | ||||
|   # The level 8 is the highest level. original was 5 | ||||
|   # 7 is more than enough, higher just leaves NULL things. | ||||
|   | ||||
| @@ -25,6 +25,7 @@ declare(strict_types=1); | ||||
| 
 | ||||
| use Rector\Config\RectorConfig; | ||||
| use Rector\EarlyReturn\Rector\If_\ChangeOrIfContinueToMultiContinueRector; | ||||
| use Rector\Php80\Rector\ClassMethod\AddParamBasedOnParentClassMethodRector; | ||||
| use Rector\Transform\Rector\String_\StringToClassConstantRector; | ||||
| use RectorLaravel\Set\LaravelLevelSetList; | ||||
| 
 | ||||
| @@ -32,6 +33,7 @@ use RectorLaravel\Set\LaravelLevelSetList; | ||||
| return RectorConfig::configure() | ||||
|     ->withSkip([ | ||||
|         ChangeOrIfContinueToMultiContinueRector::class, | ||||
|         AddParamBasedOnParentClassMethodRector::class, | ||||
|         StringToClassConstantRector::class => [ | ||||
|             __DIR__ . '/../app/Http/Controllers/Auth/LoginController.php', | ||||
|         ], | ||||
|   | ||||
| @@ -314,8 +314,9 @@ DEMO_USERNAME= | ||||
| DEMO_PASSWORD= | ||||
|  | ||||
| # | ||||
| # Disable or enable the running balance column data | ||||
| # Please disable this. It's a very experimental feature. | ||||
| # Disable or enable the running balance column data. | ||||
| # If you enable this, please also run "php artisan firefly-iii:correct-database" | ||||
| # This will take some time the first run. | ||||
| # | ||||
| USE_RUNNING_BALANCE=false | ||||
|  | ||||
| @@ -329,7 +330,7 @@ FIREFLY_III_LAYOUT=v1 | ||||
| # Which Query Parser implementation to use for the search engine and rules | ||||
| # 'new' is experimental, 'legacy' is the classic one | ||||
| # | ||||
| QUERY_PARSER_IMPLEMENTATION=legacy | ||||
| QUERY_PARSER_IMPLEMENTATION=new | ||||
|  | ||||
| # | ||||
| # Please make sure this URL matches the external URL of your Firefly III installation. | ||||
|   | ||||
							
								
								
									
										13
									
								
								.github/ISSUE_TEMPLATE/bug.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										13
									
								
								.github/ISSUE_TEMPLATE/bug.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| name: Bug Report | ||||
| name: Bug report | ||||
| description: Report a bug in Firefly III (or associated tools) | ||||
| body: | ||||
|   - type: checkboxes | ||||
| @@ -12,7 +12,7 @@ body: | ||||
|   - type: checkboxes | ||||
|     attributes: | ||||
|       label: I've found a bug and checked that ... | ||||
|       description: Make sure that your request fulfills all of the following requirements. If one requirement cannot be satisfied, explain in detail why. | ||||
|       description: Make sure that your request fulfills all of the following requirements. If one requirement cannot be satisfied, please explain why. | ||||
|       options: | ||||
|         - label: ... [the documentation](https://docs.firefly-iii.org/) does not mention anything about my problem | ||||
|         - label: ... there are no open or closed issues that are related to my problem | ||||
| @@ -33,13 +33,6 @@ body: | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: Expected behaviour | ||||
|       description: Please describe precisely what you'd expect to happen. Be specific. | ||||
|     validations: | ||||
|       required: false | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: Steps to reproduce | ||||
| @@ -54,4 +47,4 @@ body: | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: Additional info | ||||
|       description: Please provide any additional information that seem useful. | ||||
|       description: Please provide any additional information that seems useful. | ||||
|   | ||||
							
								
								
									
										3
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							| @@ -3,3 +3,6 @@ contact_links: | ||||
|   - name: Ask a question | ||||
|     url: https://github.com/firefly-iii/firefly-iii/discussions | ||||
|     about: Please ask and answer questions here. | ||||
|   - name: I need support!  | ||||
|     url: https://github.com/firefly-iii/firefly-iii/discussions | ||||
|     about: I think I broke something... | ||||
|   | ||||
							
								
								
									
										7
									
								
								.github/ISSUE_TEMPLATE/fr.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.github/ISSUE_TEMPLATE/fr.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,4 @@ | ||||
| name: Feature Request | ||||
| name: Feature request | ||||
| description: Request a feature or enhancement in Firefly III (or associated tools) | ||||
| body: | ||||
|   - type: checkboxes | ||||
| @@ -31,11 +31,6 @@ body: | ||||
|     validations: | ||||
|       required: true | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: What are alternatives? | ||||
|       description: Please describe what alternatives currently exist. | ||||
|  | ||||
|   - type: textarea | ||||
|     attributes: | ||||
|       label: Additional context | ||||
|   | ||||
							
								
								
									
										6
									
								
								.github/label-actions.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.github/label-actions.yml
									
									
									
									
										vendored
									
									
								
							| @@ -25,7 +25,7 @@ feature: | ||||
|  | ||||
|       This issue has been marked as a feature request. | ||||
|  | ||||
|       If you come across this issue, please be aware there is NO need to reply with "+1" or "me too" or "I need this too" or whatever. Such comments are not helpful, and do not influence [the roadmap](https://roadmap.firefly-iii.org/). Your comment may be :skull: deleted. You can subscribe to this issue to get updates. | ||||
|       If you come across this issue, please be aware there is NO need to reply with "+1" or "I need this too" or "any updates?" or whatever. Such comments are not helpful, and do not influence [the roadmap](https://roadmap.firefly-iii.org/). Your comment may be :skull: deleted. You can subscribe to this issue to get updates. | ||||
|  | ||||
|       Thank you for your contributions. | ||||
|  | ||||
| @@ -39,7 +39,7 @@ epic: | ||||
|  | ||||
|       This issue has been marked as an epic. In epics, large amounts of works are collected that will be part of a major new feature. If you have more ideas that could be a part of this epic, feel free to reply. | ||||
|  | ||||
|       *However*, please be aware there is NO need to reply with "+1" or "me too" or "I need this too" or whatever. Such comments are not helpful, and do not influence [the roadmap](https://roadmap.firefly-iii.org/). Your comment may be :skull: deleted. | ||||
|       *However*, please be aware there is NO need to reply with "+1" or "I need this too" or "any updates?" or whatever. Such comments are not helpful, and do not influence [the roadmap](https://roadmap.firefly-iii.org/). Your comment may be :skull: deleted. | ||||
|  | ||||
|       If you are merely interested in this epic's progress, you can subscribe to this issue to get updates. | ||||
|  | ||||
| @@ -56,7 +56,7 @@ enhancement: | ||||
|  | ||||
|       This issue has been marked as an enhancement. | ||||
|  | ||||
|       If you come across this issue, please be aware there is NO need to reply with "+1" or "me too" or "I need this too" or whatever. Such comments are not helpful, and do not influence [the roadmap](https://roadmap.firefly-iii.org/). Your comment may be :skull: deleted. You can subscribe to this issue to get updates. | ||||
|       If you come across this issue, please be aware there is NO need to reply with "+1" or "I need this too" or "any updates?" or whatever. Such comments are not helpful, and do not influence [the roadmap](https://roadmap.firefly-iii.org/). Your comment may be :skull: deleted. You can subscribe to this issue to get updates. | ||||
|  | ||||
|       Thank you for your contributions. | ||||
|  | ||||
|   | ||||
							
								
								
									
										25
									
								
								.github/release-notes/alpha.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								.github/release-notes/alpha.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| Welcome to release %version of Firefly III. This **alpha** release contains the latest fixes, translations and features. It is probably buggy and may not work as expected. You can download the release below, and adventurous Docker users can find this release under the `alpha` tag. | ||||
| 
 | ||||
| > [!WARNING] | ||||
| > Please be careful with this alpha release, as it may not work as expected. | ||||
| 
 | ||||
| Alpha releases are created to test new features and fixes before they are included in a stable release. They are not recommended for production use. This release was created on %date and may contain unexpected bugs. Data loss is rare but possible. | ||||
| 
 | ||||
| ## Changelog (not final) | ||||
| 
 | ||||
| %changelog | ||||
| 
 | ||||
| ## Installation and upgrade instructions | ||||
| 
 | ||||
| * Please read the installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/). | ||||
| * Alternatively, read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/) | ||||
| 
 | ||||
| The release files are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/). | ||||
| 
 | ||||
| ## Develop with Firefly III | ||||
| 
 | ||||
| Are you interested in (future) API changes to Firefly III, or other interesting dev-related updates? Sign up to the [Firefly III developer newsletter](https://firefly-iii.kit.com/dev) to receive low-frequency updates about the development of Firefly III. | ||||
| 
 | ||||
| ## Support Firefly III | ||||
| 
 | ||||
| Did you know you can support the development of Firefly III? You can donate in many ways, like GitHub Sponsors or Patreon. Please [follow this link](https://bit.ly/donate-to-Firefly-III) for more information. Thank you for your consideration. | ||||
							
								
								
									
										25
									
								
								.github/release-notes/beta.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								.github/release-notes/beta.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| Welcome to release %version of Firefly III. This **beta** release contains the latest fixes, translations and features. It may be buggy, nor work as expected. You can download the release below, and adventurous Docker users can find this release under the `beta` tag. | ||||
| 
 | ||||
| > [!WARNING] | ||||
| > Please be careful with this beta release, as it may not work as expected. | ||||
| 
 | ||||
| Alpha releases are created to test new features and fixes before they are included in a stable release. They are not recommended for production use. This release was created on %date and may contain unexpected bugs. Data loss is rare but possible. | ||||
| 
 | ||||
| ## Changelog (not final) | ||||
| 
 | ||||
| %changelog | ||||
| 
 | ||||
| ## Installation and upgrade instructions | ||||
| 
 | ||||
| * Please read the installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/). | ||||
| * Alternatively, read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/) | ||||
| 
 | ||||
| The release files are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/). | ||||
| 
 | ||||
| ## Develop with Firefly III | ||||
| 
 | ||||
| Are you interested in (future) API changes to Firefly III, or other interesting dev-related updates? Sign up to the [Firefly III developer newsletter](https://firefly-iii.kit.com/dev) to receive low-frequency updates about the development of Firefly III. | ||||
| 
 | ||||
| ## Support Firefly III | ||||
| 
 | ||||
| Did you know you can support the development of Firefly III? You can donate in many ways, like GitHub Sponsors or Patreon. Please [follow this link](https://bit.ly/donate-to-Firefly-III) for more information. Thank you for your consideration. | ||||
							
								
								
									
										25
									
								
								.github/release-notes/branch.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								.github/release-notes/branch.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| Welcome to release %version of Firefly III. This branch-related release contains the latest fixes, translations and features. It is probably buggy and may not work as expected. You can download the release below, and adventurous Docker users can find this release under the `branch-*` tag. | ||||
| 
 | ||||
| > [!WARNING] | ||||
| > Please be careful with this branch release, as it may not work as expected. | ||||
| 
 | ||||
| Branch releases are created to test large new features that are developed alongside the normal release flow. They are not recommended for production use. This release was created on %date and may contain unexpected bugs. Data loss is rare but possible. | ||||
| 
 | ||||
| ## Changelog | ||||
| 
 | ||||
| There is no changelog for this release, as it is not final. However, [changelog.md](https://github.com/firefly-iii/firefly-iii/blob/develop/changelog.md) may already contain entries for the future release that this branch will be a part of. | ||||
| 
 | ||||
| ## Installation and upgrade instructions | ||||
| 
 | ||||
| * Please read the installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/). | ||||
| * Alternatively, read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/) | ||||
| 
 | ||||
| The release files are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/). | ||||
| 
 | ||||
| ## Develop with Firefly III | ||||
| 
 | ||||
| Are you interested in (future) API changes to Firefly III, or other interesting dev-related updates? Sign up to the [Firefly III developer newsletter](https://firefly-iii.kit.com/dev) to receive low-frequency updates about the development of Firefly III. | ||||
| 
 | ||||
| ## Support Firefly III | ||||
| 
 | ||||
| Did you know you can support the development of Firefly III? You can donate in many ways, like GitHub Sponsors or Patreon. Please [follow this link](https://bit.ly/donate-to-Firefly-III) for more information. Thank you for your consideration. | ||||
							
								
								
									
										25
									
								
								.github/release-notes/develop.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								.github/release-notes/develop.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,25 @@ | ||||
| Welcome to the latest development release of Firefly III. This test release contains the absolute latest fixes, translations and features. It is probably buggy and may not work as expected. You can download the release below, and adventurous Docker users can find this release under the `develop` tag. | ||||
| 
 | ||||
| > [!WARNING] | ||||
| > Please be careful with this pre-release, as it may not work as expected. | ||||
| 
 | ||||
| This release was created on %date and may contain unexpected bugs. Data loss is rare but possible. | ||||
| 
 | ||||
| ## Changelog | ||||
| 
 | ||||
| The changelog for this release may not be up-to-date, so it is not included. However, [changelog.md](https://github.com/firefly-iii/firefly-iii/blob/develop/changelog.md) may already contain entries for the future release. | ||||
| 
 | ||||
| ## Installation and upgrade instructions | ||||
| 
 | ||||
| * Please read the installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/). | ||||
| * Alternatively, read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/) | ||||
| 
 | ||||
| The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/). | ||||
| 
 | ||||
| ## Develop with Firefly III | ||||
| 
 | ||||
| Are you interested in (future) API changes to Firefly III, or other interesting dev-related updates? Sign up to the [Firefly III developer newsletter](https://firefly-iii.kit.com/dev) to receive low-frequency updates about the development of Firefly III. | ||||
| 
 | ||||
| ## Support Firefly III | ||||
| 
 | ||||
| Did you know you can support the development of Firefly III? You can donate in many ways, like GitHub Sponsors or Patreon. Please [follow this link](https://bit.ly/donate-to-Firefly-III) for more information. Thank you for your consideration. | ||||
							
								
								
									
										20
									
								
								.github/release-notes/release.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								.github/release-notes/release.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| Welcome to release %version of Firefly III. It contains the latest fixes, translations and features. Docker users can find this release under the `latest` tag. | ||||
| 
 | ||||
| ## Changelog | ||||
| 
 | ||||
| %changelog | ||||
| 
 | ||||
| ## Installation and upgrade instructions | ||||
| 
 | ||||
| * Please read the installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/). | ||||
| * Alternatively, read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/) | ||||
| 
 | ||||
| The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/). | ||||
| 
 | ||||
| ## Develop with Firefly III | ||||
| 
 | ||||
| Are you interested in (future) API changes to Firefly III, or other interesting dev-related updates? Sign up to the [Firefly III developer newsletter](https://firefly-iii.kit.com/dev) to receive low-frequency updates about the development of Firefly III. | ||||
| 
 | ||||
| ## Support Firefly III | ||||
| 
 | ||||
| Did you know you can support the development of Firefly III? You can donate in many ways, like GitHub Sponsors or Patreon. Please [follow this link](https://bit.ly/donate-to-Firefly-III) for more information. Thank you for your consideration. | ||||
							
								
								
									
										5
									
								
								.github/workflows/cleanup.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.github/workflows/cleanup.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,7 +15,7 @@ jobs: | ||||
|     timeout-minutes: 10 | ||||
|     steps: | ||||
|       - name: Prune cancelled/skipped runs | ||||
|         uses: actions/github-script@v7 | ||||
|         uses: actions/github-script@v8 | ||||
|         with: | ||||
|           github-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           script: | | ||||
| @@ -45,7 +45,7 @@ jobs: | ||||
|             } | ||||
|  | ||||
|       - name: Prune runs older than 3 days | ||||
|         uses: actions/github-script@v7 | ||||
|         uses: actions/github-script@v8 | ||||
|         with: | ||||
|           github-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           script: | | ||||
| @@ -66,7 +66,6 @@ jobs: | ||||
|               'label-actions.yml', | ||||
|               'lock.yml', | ||||
|               'release.yml', | ||||
|               'sonarcloud.yml', | ||||
|               'stale.yml' | ||||
|             ] | ||||
|  | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/close-duplicates.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/close-duplicates.yml
									
									
									
									
										vendored
									
									
								
							| @@ -13,7 +13,7 @@ jobs: | ||||
|   close_duplicates: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: github/command@v2.0.1 | ||||
|       - uses: github/command@v2.0.2 | ||||
|         id: command | ||||
|         with: | ||||
|           allowed_contexts: "issue" | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/depsreview.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/depsreview.yml
									
									
									
									
										vendored
									
									
								
							| @@ -9,7 +9,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: 'Checkout repository' | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@v5 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: 'Dependency review' | ||||
|   | ||||
							
								
								
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/lock.yml
									
									
									
									
										vendored
									
									
								
							| @@ -21,7 +21,7 @@ jobs: | ||||
|       discussions: write | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: dessant/lock-threads@v5 | ||||
|       - uses: JC5/lock-threads@v6.0.6 | ||||
|         with: | ||||
|           issue-inactive-days: 21 | ||||
|           pr-inactive-days: 21 | ||||
|   | ||||
							
								
								
									
										112
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										112
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
								
							| @@ -19,7 +19,7 @@ jobs: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|         uses: actions/checkout@v5 | ||||
|         with: | ||||
|           fetch-depth: 0 | ||||
|       - name: Import GPG key | ||||
| @@ -50,7 +50,7 @@ jobs: | ||||
|           git pull | ||||
|           echo "Current branch is $(git branch --show-current)" | ||||
|         env: | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }} | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || inputs.version }} | ||||
|       - name: Configure Git | ||||
|         run: | | ||||
|           # do some configuration | ||||
| @@ -118,7 +118,7 @@ jobs: | ||||
|         env: | ||||
|           FIREFLY_III_ROOT: /github/workspace | ||||
|           GH_TOKEN: "" | ||||
|           FF_III_VERSION: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }} | ||||
|           FF_III_VERSION: ${{ github.event_name == 'schedule' && 'develop' || inputs.version }} | ||||
|       - name: Generate JSON v1 | ||||
|         id: json-v1 | ||||
|         uses: JC5/firefly-iii-dev@main | ||||
| @@ -221,7 +221,7 @@ jobs: | ||||
|           echo "tarName=$tarName" >> "$GITHUB_ENV" | ||||
|           echo "BRANCH_NAME=$BRANCH_NAME" >> "$GITHUB_ENV" | ||||
|         env: | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }} | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || inputs.version }} | ||||
|       - name: Commit all changes | ||||
|         run: | | ||||
|           # add all content, except output.txt (this contains the changelog and/or the download instructions) | ||||
| @@ -232,102 +232,16 @@ jobs: | ||||
|           git commit -m "🤖 Auto commit for release '$version' on $(date +'%Y-%m-%d')" || true | ||||
|           git push | ||||
|         env: | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }} | ||||
|       - name: Extract changelog | ||||
|         id: extract-changelog | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || inputs.version }} | ||||
|       - name: Generate release description | ||||
|         id: release-description | ||||
|         uses: JC5/firefly-iii-dev@main | ||||
|         with: | ||||
|           action: 'ff3:extract-changelog' | ||||
|           action: "ff3:generate-release-notes firefly-iii ${{ inputs.version || 'develop' }}" | ||||
|           output: 'output' | ||||
|         env: | ||||
|           FIREFLY_III_ROOT: /github/workspace | ||||
|           GH_TOKEN: "" | ||||
|       - name: Describe new release | ||||
|         run: | | ||||
|  | ||||
|           # describe the development release. | ||||
|           if [[ "develop" == "$version" ]]; then | ||||
|             echo 'Describe the latest develop release' | ||||
|             rm -f output.txt | ||||
|             touch output.txt | ||||
|             sudo chown -R runner:docker output.txt | ||||
|             echo "Weekly development release of Firefly III with the latest fixes, translations and features. Docker users can find this release under the \`develop\` tag." >> output.txt | ||||
|             echo "" >> output.txt | ||||
|             echo "This release was created on **$(date +'%Y-%m-%d %H:%M')** and may contain unexpected bugs. Data loss is rare but is not impossible. The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/)." >> output.txt | ||||
|             echo "" >> output.txt | ||||
|             echo "* Please read the installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/)" >> output.txt | ||||
|             echo "* Or read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/)" >> output.txt | ||||
|             echo "" >> output.txt | ||||
|             echo ":warning: Please be careful with this pre-release, as it may not work as expected." >> output.txt | ||||
|           fi | ||||
|           # describe a branch release | ||||
|           if [[ "$version" == branch* ]]; then | ||||
|             echo 'Describe a branch release' | ||||
|             rm -f output.txt | ||||
|             touch output.txt | ||||
|             sudo chown -R runner:docker output.txt | ||||
|             echo "Irregular BRANCH release of Firefly III. This release contains specific features or changes. Docker users can find this release under the \`$version\` tag." >> output.txt | ||||
|             echo "" >> output.txt | ||||
|             echo "This release was created on **$(date +'%Y-%m-%d %H:%M')** and may contain unexpected bugs. Data loss is rare but is not impossible. The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/)." >> output.txt | ||||
|             echo "" >> output.txt | ||||
|             echo "* Please read the installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/)" >> output.txt | ||||
|             echo "* Or read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/)" >> output.txt | ||||
|             echo "" >> output.txt | ||||
|             echo ":warning: Please be careful with this branch pre-release, as it may not work as expected." >> output.txt | ||||
|           fi | ||||
|           # describe the main release | ||||
|           if [[ "develop" != "$version" ]] && [[ "$version" != branch* ]] && [[ "$version" != *alpha* ]] && [[ "$version" != *beta* ]]; then | ||||
|             echo 'Describe the latest release' | ||||
|             sudo chown -R runner:docker output.txt | ||||
|             touch output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo "Welcome to release $version of Firefly III. It contains the the latest fixes, translations and features. Docker users can find this release under the \`latest\` tag." >> output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo '### Instructions' >> output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo "* Installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/)" >> output.txt | ||||
|             echo "* Or read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/)" >> output.txt | ||||
|             echo "* The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/)." >> output.txt | ||||
|  | ||||
|           fi | ||||
|  | ||||
|           # describe alpha release | ||||
|           if [[ "$version" == *alpha* ]]; then | ||||
|             echo 'Describe an ALPHA release' | ||||
|             rm -f output.txt | ||||
|             touch output.txt | ||||
|             sudo chown -R runner:docker output.txt | ||||
|             echo "Very early ALPHA release of Firefly III. This release contains specific features or changes. Docker users can find this release under the \`$version\` tag." >> output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo "This release was created on **$(date +'%Y-%m-%d %H:%M')** and may contain unexpected bugs. Data loss is rare but is not impossible. The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/)." >> output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo '### Instructions' >> output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo "* Installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/)" >> output.txt | ||||
|             echo "* Or read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/)" >> output.txt | ||||
|             echo "* The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/)." >> output.txt | ||||
|  | ||||
|           fi | ||||
|  | ||||
|           # describe beta release | ||||
|           if [[ "$version" == *beta* ]]; then | ||||
|             echo 'Describe a BETA release' | ||||
|             rm -f output.txt | ||||
|             touch output.txt | ||||
|             sudo chown -R runner:docker output.txt | ||||
|             echo "Very early BETA release of Firefly III. This release contains specific features or changes. Docker users can find this release under the \`$version\` tag." >> output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo "This release was created on **$(date +'%Y-%m-%d %H:%M')** and may contain unexpected bugs. Data loss is rare but is not impossible. The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/)." >> output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo '### Instructions' >> output.txt | ||||
|             echo '' >> output.txt | ||||
|             echo "* Installation instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/installation/docker/), [Portainer](https://docs.firefly-iii.org/how-to/firefly-iii/installation/portainer/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/installation/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/installation/self-managed/)" >> output.txt | ||||
|             echo "* Or read the upgrade instructions for [Docker](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/docker/), [Kubernetes](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/kubernetes/) or [self-managed servers](https://docs.firefly-iii.org/how-to/firefly-iii/upgrade/self-managed/)" >> output.txt | ||||
|             echo "* The releases are signed, and you can verify them using the [Firefly III releases PGP key](https://docs.firefly-iii.org/explanation/more-information/signatures/)." >> output.txt | ||||
|  | ||||
|           fi | ||||
|         env: | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }} | ||||
|       - name: Merge all into working branch | ||||
|         run: | | ||||
|           MERGE_INTO=develop | ||||
| @@ -336,7 +250,7 @@ jobs: | ||||
|             fi | ||||
|  | ||||
|           echo "Merge all changes from $BRANCH_NAME back into '$MERGE_INTO' using a PR" | ||||
|           PR_URL=$(gh pr create -B $MERGE_INTO -H $BRANCH_NAME --title "🤖 Automatic PR to merge all changes into the '$MERGE_INTO' branch." --body '🤖 Created by GitHub action') | ||||
|           PR_URL=$(gh pr create -B $MERGE_INTO -H $BRANCH_NAME --title "🤖 Automatic PR to merge all changes into the '$MERGE_INTO' branch." --body '🤖 This PR was created automatically by a GitHub action to merge the changed files into this branch. It will be merged automatically. `Share and enjoy`') | ||||
|           echo "PR URL is '$PR_URL'" | ||||
|           IFS='/' read -ra parts <<< "$PR_URL" | ||||
|           PR_NR=$(printf %s\\n "${parts[@]:(-1)}") | ||||
| @@ -358,7 +272,7 @@ jobs: | ||||
|  | ||||
|             echo "Also merge everything into main since this is a release." | ||||
|             echo 'create PR' | ||||
|             PR_URL=$(gh pr create -B main -H develop --title "🤖 Automatic PR to merge all changes into the main branch." --body "🤖 Created by GitHub action") | ||||
|             PR_URL=$(gh pr create -B main -H develop --title "🤖 Automatic PR to merge all changes into the main branch." --body '🤖 This PR was created automatically by a GitHub action to merge the changed files into this branch. It will be merged automatically. `Share and enjoy`') | ||||
|             echo "PR URL is '$PR_URL'" | ||||
|  | ||||
|             IFS='/' read -ra parts <<< "$PR_URL" | ||||
| @@ -377,7 +291,7 @@ jobs: | ||||
|           echo "DONE!" | ||||
|         env: | ||||
|           GH_TOKEN: ${{ github.token }} | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }} | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || inputs.version }} | ||||
|       - name: Create archives | ||||
|         run: | | ||||
|           echo "Create zip file $zipName" | ||||
| @@ -461,7 +375,7 @@ jobs: | ||||
|           fi | ||||
|         env: | ||||
|           GH_TOKEN: ${{ github.token }} | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }} | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || inputs.version }} | ||||
|       - name: Upload artifacts | ||||
|         run: | | ||||
|           # add zip file to release. | ||||
| @@ -497,4 +411,4 @@ jobs: | ||||
|           rm -f $tarName.sha256 | ||||
|         env: | ||||
|           GH_TOKEN: ${{ github.token }} | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }} | ||||
|           version: ${{ github.event_name == 'schedule' && 'develop' || inputs.version }} | ||||
|   | ||||
							
								
								
									
										71
									
								
								.github/workflows/sonarcloud.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										71
									
								
								.github/workflows/sonarcloud.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,71 +0,0 @@ | ||||
| name: 'Code - Run Sonarcloud' | ||||
| on: | ||||
|   pull_request: | ||||
|   workflow_dispatch: | ||||
|   push: | ||||
|     branches: | ||||
|       - main | ||||
| env: | ||||
|   DB_CONNECTION: sqlite | ||||
|   APP_KEY: TestTestTestTestTestTestTestTest | ||||
| jobs: | ||||
|   sonarcloud: | ||||
|     name: SonarCloud | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Setup PHP with Xdebug | ||||
|         uses: shivammathur/setup-php@v2 | ||||
|         with: | ||||
|           php-version: '8.4' | ||||
|           coverage: xdebug | ||||
|           extensions: >- | ||||
|             bcmath | ||||
|             curl | ||||
|             fileinfo | ||||
|             iconv | ||||
|             intl | ||||
|             json | ||||
|             sqlite3 | ||||
|             mbstring | ||||
|             openssl | ||||
|             pdo | ||||
|             session | ||||
|             simplexml | ||||
|             sodium | ||||
|             tokenizer | ||||
|             xml | ||||
|             xmlwriter | ||||
|  | ||||
|       - name: Copy standard configuration | ||||
|         run: cp .env.testing .env | ||||
|  | ||||
|       - name: Install Composer dependencies | ||||
|         run: composer install --prefer-dist --no-interaction --no-progress --no-scripts | ||||
|  | ||||
|       - name: "Create database file" | ||||
|         run: | | ||||
|           touch storage/database/database.sqlite | ||||
|           wget -q https://github.com/firefly-iii/test-fixtures/raw/refs/heads/main/test-database.sqlite -O storage/database/database.sqlite | ||||
|  | ||||
|       - name: "Upgrades the database to the latest version" | ||||
|         run: | | ||||
|           php artisan firefly-iii:upgrade-database | ||||
|           chmod 600 storage/oauth-public.key | ||||
|           chmod 600 storage/oauth-private.key | ||||
|  | ||||
|       - name: "Integrity Database Report" | ||||
|         run: php artisan firefly-iii:report-integrity | ||||
|  | ||||
|       - name: "Run tests with coverage" | ||||
|         run: composer coverage | ||||
|  | ||||
|       - name: Fix code coverage paths | ||||
|         run: sed -i 's@'$GITHUB_WORKSPACE'@/github/workspace/@g' coverage.xml | ||||
|  | ||||
|       - name: SonarCloud Scan | ||||
|         uses: SonarSource/sonarqube-scan-action@v5.2.0 | ||||
|         env: | ||||
|           GITHUB_TOKEN: ${{ secrets.GH_ACTIONS_PERSONAL_ACCESS_TOKEN }} | ||||
|           SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }} | ||||
							
								
								
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/stale.yml
									
									
									
									
										vendored
									
									
								
							| @@ -15,7 +15,7 @@ jobs: | ||||
|       actions: write | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/stale@v9 | ||||
|       - uses: actions/stale@v10 | ||||
|         with: | ||||
|           repo-token: ${{ secrets.GITHUB_TOKEN }} | ||||
|           stale-issue-message: | | ||||
|   | ||||
| @@ -4,6 +4,7 @@ Over time, many people have contributed to Firefly III. Their efforts are not al | ||||
| Please find below all the people who contributed to the Firefly III code. Their names are mentioned in the year of their first contribution. | ||||
| 
 | ||||
| ## 2025 | ||||
| - Nicky De Maeyer | ||||
| - Denis Iskandarov | ||||
| - = | ||||
| - Lompi | ||||
|   | ||||
| @@ -27,13 +27,14 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\AccountTypeEnum; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Account; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Support\Debug\Timer; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use FireflyIII\Support\Http\Api\AccountFilter; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Support\Facades\Log; | ||||
| 
 | ||||
| @@ -45,7 +46,8 @@ class AccountController extends Controller | ||||
|     use AccountFilter; | ||||
| 
 | ||||
|     // this array only exists to test if the constructor will use it properly.
 | ||||
|     protected array $accepts = ['application/json', 'application/vnd.api+json']; | ||||
|     protected array $accepts       = ['application/json', 'application/vnd.api+json']; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     /** @var array<int, string> */ | ||||
|     private array                      $balanceTypes; | ||||
| @@ -59,10 +61,10 @@ class AccountController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user             = auth()->user(); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(AccountRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -79,40 +81,40 @@ class AccountController extends Controller | ||||
|      */ | ||||
|     public function accounts(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data   = $request->getData(); | ||||
|         $types  = $data['types']; | ||||
|         $query  = $data['query']; | ||||
|         $date   = $data['date'] ?? today(config('app.timezone')); | ||||
|         $return = []; | ||||
|         Timer::start(sprintf('AC accounts "%s"', $query)); | ||||
|         $result = $this->repository->searchAccount((string) $query, $types, $this->parameters->get('limit')); | ||||
|         $data        = $request->getData(); | ||||
|         $types       = $data['types']; | ||||
|         $query       = $data['query']; | ||||
|         $date        = $data['date'] ?? today(config('app.timezone')); | ||||
|         $return      = []; | ||||
|         $timer       = Timer::getInstance(); | ||||
|         $timer->start(sprintf('AC accounts "%s"', $query)); | ||||
|         $result      = $this->repository->searchAccount((string) $query, $types, $this->parameters->get('limit')); | ||||
| 
 | ||||
|         // set date to subday + end-of-day for account balance. so it is at $date 23:59:59
 | ||||
|         $date->endOfDay(); | ||||
| 
 | ||||
|         $allBalances = Steam::accountsBalancesOptimized($result, $date, $this->primaryCurrency, $this->convertToPrimary); | ||||
| 
 | ||||
|         /** @var Account $account */ | ||||
|         foreach ($result as $account) { | ||||
|             $nameWithBalance = $account->name; | ||||
|             $currency        = $this->repository->getAccountCurrency($account) ?? $this->nativeCurrency; | ||||
|             $currency        = $this->repository->getAccountCurrency($account) ?? $this->primaryCurrency; | ||||
|             $useCurrency     = $currency; | ||||
|             if (in_array($account->accountType->type, $this->balanceTypes, true)) { | ||||
|                 // this one is correct.
 | ||||
|                 Log::debug(sprintf('accounts: Call finalAccountBalance with date/time "%s"', $date->toIso8601String())); | ||||
|                 $balance         = Steam::finalAccountBalance($account, $date); | ||||
|                 $key             = $this->convertToNative && $currency->id !== $this->nativeCurrency->id ? 'native_balance' : 'balance'; | ||||
|                 $useCurrency     = $this->convertToNative && $currency->id !== $this->nativeCurrency->id ? $this->nativeCurrency : $currency; | ||||
|                 $balance         = $allBalances[$account->id] ?? []; | ||||
|                 $key             = $this->convertToPrimary && $currency->id !== $this->primaryCurrency->id ? 'pc_balance' : 'balance'; | ||||
|                 $useCurrency     = $this->convertToPrimary && $currency->id !== $this->primaryCurrency->id ? $this->primaryCurrency : $currency; | ||||
|                 $amount          = $balance[$key] ?? '0'; | ||||
|                 $nameWithBalance = sprintf( | ||||
|                     '%s (%s)', | ||||
|                     $account->name, | ||||
|                     app('amount')->formatAnything($useCurrency, $amount, false) | ||||
|                 ); | ||||
|                 $nameWithBalance = sprintf('%s (%s)', $account->name, Amount::formatAnything($useCurrency, $amount, false)); | ||||
|             } | ||||
| 
 | ||||
|             $return[]        = [ | ||||
|                 'id'                              => (string) $account->id, | ||||
|                 'name'                            => $account->name, | ||||
|                 'name_with_balance'               => $nameWithBalance, | ||||
|                 'active'                          => $account->active, | ||||
|                 'type'                            => $account->accountType->type, | ||||
|                 'currency_id'                     => (string) $useCurrency->id, | ||||
|                 'currency_name'                   => $useCurrency->name, | ||||
| @@ -138,7 +140,7 @@ class AccountController extends Controller | ||||
|                 return $posA - $posB; | ||||
|             } | ||||
|         ); | ||||
|         Timer::stop(sprintf('AC accounts "%s"', $query)); | ||||
|         $timer->stop(sprintf('AC accounts "%s"', $query)); | ||||
| 
 | ||||
|         return response()->api($return); | ||||
|     } | ||||
|   | ||||
| @@ -26,9 +26,9 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\Bill; | ||||
| use FireflyIII\Repositories\Bill\BillRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -37,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class BillController extends Controller | ||||
| { | ||||
|     private BillRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_SUBSCRIPTIONS]; | ||||
| 
 | ||||
|     /** | ||||
|      * BillController constructor. | ||||
| @@ -46,10 +47,10 @@ class BillController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user             = auth()->user(); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(BillRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|   | ||||
| @@ -26,9 +26,9 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\Budget; | ||||
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -37,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class BudgetController extends Controller | ||||
| { | ||||
|     private BudgetRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_BUDGETS]; | ||||
| 
 | ||||
|     /** | ||||
|      * BudgetController constructor. | ||||
| @@ -46,10 +47,10 @@ class BudgetController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user             = auth()->user(); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(BudgetRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -66,8 +67,9 @@ class BudgetController extends Controller | ||||
|         $result   = $this->repository->searchBudget($data['query'], $this->parameters->get('limit')); | ||||
|         $filtered = $result->map( | ||||
|             static fn (Budget $item) => [ | ||||
|                 'id'   => (string) $item->id, | ||||
|                 'name' => $item->name, | ||||
|                 'id'     => (string) $item->id, | ||||
|                 'name'   => $item->name, | ||||
|                 'active' => $item->active, | ||||
|             ] | ||||
|         ); | ||||
| 
 | ||||
|   | ||||
| @@ -26,9 +26,9 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\Category; | ||||
| use FireflyIII\Repositories\Category\CategoryRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -37,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class CategoryController extends Controller | ||||
| { | ||||
|     private CategoryRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     /** | ||||
|      * CategoryController constructor. | ||||
| @@ -46,10 +47,10 @@ class CategoryController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user             = auth()->user(); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(CategoryRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|   | ||||
| @@ -24,11 +24,12 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use Deprecated; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\Currency\CurrencyRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -37,6 +38,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class CurrencyController extends Controller | ||||
| { | ||||
|     private CurrencyRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     /** | ||||
|      * CurrencyController constructor. | ||||
| @@ -46,10 +48,10 @@ class CurrencyController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user             = auth()->user(); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(CurrencyRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -83,9 +85,8 @@ class CurrencyController extends Controller | ||||
|     /** | ||||
|      * Documentation for this endpoint is at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getCurrenciesCodeAC
 | ||||
|      * | ||||
|      * @deprecated | ||||
|      */ | ||||
|     #[Deprecated]
 | ||||
|     public function currenciesWithCode(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data       = $request->getData(); | ||||
|   | ||||
| @@ -26,9 +26,9 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\ObjectGroup; | ||||
| use FireflyIII\Repositories\ObjectGroup\ObjectGroupRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -37,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class ObjectGroupController extends Controller | ||||
| { | ||||
|     private ObjectGroupRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     /** | ||||
|      * CurrencyController constructor. | ||||
| @@ -46,10 +47,10 @@ class ObjectGroupController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user             = auth()->user(); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(ObjectGroupRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|   | ||||
| @@ -26,10 +26,12 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\PiggyBank; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Repositories\PiggyBank\PiggyBankRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -39,6 +41,7 @@ class PiggyBankController extends Controller | ||||
| { | ||||
|     private AccountRepositoryInterface   $accountRepository; | ||||
|     private PiggyBankRepositoryInterface $piggyRepository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_PIGGY_BANKS]; | ||||
| 
 | ||||
|     /** | ||||
|      * PiggyBankController constructor. | ||||
| @@ -48,22 +51,19 @@ class PiggyBankController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user                    = auth()->user(); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->piggyRepository   = app(PiggyBankRepositoryInterface::class); | ||||
|                 $this->accountRepository = app(AccountRepositoryInterface::class); | ||||
|                 $this->piggyRepository->setUser($user); | ||||
|                 $this->accountRepository->setUser($user); | ||||
|                 $this->piggyRepository->setUser($this->user); | ||||
|                 $this->piggyRepository->setUserGroup($this->userGroup); | ||||
|                 $this->accountRepository->setUser($this->user); | ||||
|                 $this->accountRepository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getPiggiesAC
 | ||||
|      */ | ||||
|     public function piggyBanks(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data     = $request->getData(); | ||||
| @@ -90,10 +90,6 @@ class PiggyBankController extends Controller | ||||
|         return response()->api($response); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getPiggiesBalanceAC
 | ||||
|      */ | ||||
|     public function piggyBanksWithBalance(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data     = $request->getData(); | ||||
| @@ -102,6 +98,7 @@ class PiggyBankController extends Controller | ||||
| 
 | ||||
|         /** @var PiggyBank $piggy */ | ||||
|         foreach ($piggies as $piggy) { | ||||
|             /** @var TransactionCurrency $currency */ | ||||
|             $currency      = $piggy->transactionCurrency; | ||||
|             $currentAmount = $this->piggyRepository->getCurrentAmount($piggy); | ||||
|             $objectGroup   = $piggy->objectGroups()->first(); | ||||
| @@ -111,8 +108,8 @@ class PiggyBankController extends Controller | ||||
|                 'name_with_balance'       => sprintf( | ||||
|                     '%s (%s / %s)', | ||||
|                     $piggy->name, | ||||
|                     app('amount')->formatAnything($currency, $currentAmount, false), | ||||
|                     app('amount')->formatAnything($currency, $piggy->target_amount, false), | ||||
|                     Amount::formatAnything($currency, $currentAmount, false), | ||||
|                     Amount::formatAnything($currency, $piggy->target_amount, false), | ||||
|                 ), | ||||
|                 'currency_id'             => (string) $currency->id, | ||||
|                 'currency_name'           => $currency->name, | ||||
|   | ||||
| @@ -26,6 +26,7 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\Recurrence; | ||||
| use FireflyIII\Repositories\Recurring\RecurringRepositoryInterface; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -36,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class RecurrenceController extends Controller | ||||
| { | ||||
|     private RecurringRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_RECURRING]; | ||||
| 
 | ||||
|     /** | ||||
|      * RecurrenceController constructor. | ||||
| @@ -45,19 +47,16 @@ class RecurrenceController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(RecurringRepositoryInterface::class); | ||||
| 
 | ||||
|                 $this->repository->setUser(auth()->user()); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getRecurringAC
 | ||||
|      */ | ||||
|     public function recurring(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data        = $request->getData(); | ||||
| @@ -70,6 +69,7 @@ class RecurrenceController extends Controller | ||||
|                 'id'          => (string) $recurrence->id, | ||||
|                 'name'        => $recurrence->title, | ||||
|                 'description' => $recurrence->description, | ||||
|                 'active'      => $recurrence->active, | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|   | ||||
| @@ -26,6 +26,7 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\Rule; | ||||
| use FireflyIII\Repositories\Rule\RuleRepositoryInterface; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -36,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class RuleController extends Controller | ||||
| { | ||||
|     private RuleRepositoryInterface $repository; | ||||
|     protected array                 $acceptedRoles = [UserRoleEnum::READ_RULES]; | ||||
| 
 | ||||
|     /** | ||||
|      * RuleController constructor. | ||||
| @@ -45,18 +47,16 @@ class RuleController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(RuleRepositoryInterface::class); | ||||
|                 $this->repository->setUser(auth()->user()); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getRulesAC
 | ||||
|      */ | ||||
|     public function rules(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data     = $request->getData(); | ||||
| @@ -66,9 +66,10 @@ class RuleController extends Controller | ||||
|         /** @var Rule $rule */ | ||||
|         foreach ($rules as $rule) { | ||||
|             $response[] = [ | ||||
|                 'id'          => (string) $rule->id, | ||||
|                 'id'          => (string)$rule->id, | ||||
|                 'name'        => $rule->title, | ||||
|                 'description' => $rule->description, | ||||
|                 'active'      => $rule->active, | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|   | ||||
| @@ -26,6 +26,7 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\RuleGroup; | ||||
| use FireflyIII\Repositories\RuleGroup\RuleGroupRepositoryInterface; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -36,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class RuleGroupController extends Controller | ||||
| { | ||||
|     private RuleGroupRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_RULES]; | ||||
| 
 | ||||
|     /** | ||||
|      * RuleGroupController constructor. | ||||
| @@ -45,18 +47,16 @@ class RuleGroupController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(RuleGroupRepositoryInterface::class); | ||||
|                 $this->repository->setUser(auth()->user()); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getRuleGroupsAC
 | ||||
|      */ | ||||
|     public function ruleGroups(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data     = $request->getData(); | ||||
| @@ -69,6 +69,7 @@ class RuleGroupController extends Controller | ||||
|                 'id'          => (string) $group->id, | ||||
|                 'name'        => $group->title, | ||||
|                 'description' => $group->description, | ||||
|                 'active'      => $group->active, | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|   | ||||
| @@ -26,9 +26,9 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\Tag; | ||||
| use FireflyIII\Repositories\Tag\TagRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -37,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class TagController extends Controller | ||||
| { | ||||
|     private TagRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     /** | ||||
|      * TagController constructor. | ||||
| @@ -46,20 +47,16 @@ class TagController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user             = auth()->user(); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(TagRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getTagAC
 | ||||
|      */ | ||||
|     public function tags(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data   = $request->getData(); | ||||
|   | ||||
| @@ -31,7 +31,6 @@ use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\TransactionJournal; | ||||
| use FireflyIII\Repositories\Journal\JournalRepositoryInterface; | ||||
| use FireflyIII\Repositories\TransactionGroup\TransactionGroupRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Support\Collection; | ||||
| 
 | ||||
| @@ -52,24 +51,19 @@ class TransactionController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user                  = auth()->user(); | ||||
|                 $userGroup             = $this->validateUserGroup($request); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository      = app(JournalRepositoryInterface::class); | ||||
|                 $this->groupRepository = app(TransactionGroupRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->groupRepository->setUser($user); | ||||
|                 $this->groupRepository->setUserGroup($userGroup); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
|                 $this->groupRepository->setUser($this->user); | ||||
|                 $this->groupRepository->setUserGroup($this->userGroup); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getTransactionsAC
 | ||||
|      */ | ||||
|     public function transactions(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data     = $request->getData(); | ||||
| @@ -92,10 +86,6 @@ class TransactionController extends Controller | ||||
|         return response()->api($array); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getTransactionsIDAC
 | ||||
|      */ | ||||
|     public function transactionsWithID(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data   = $request->getData(); | ||||
|   | ||||
| @@ -26,6 +26,7 @@ namespace FireflyIII\Api\V1\Controllers\Autocomplete; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Autocomplete\AutocompleteRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\TransactionType; | ||||
| use FireflyIII\Repositories\TransactionType\TransactionTypeRepositoryInterface; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -36,6 +37,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class TransactionTypeController extends Controller | ||||
| { | ||||
|     private TransactionTypeRepositoryInterface $repository; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     /** | ||||
|      * TransactionTypeController constructor. | ||||
| @@ -45,6 +47,7 @@ class TransactionTypeController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(TransactionTypeRepositoryInterface::class); | ||||
| 
 | ||||
|                 return $next($request); | ||||
| @@ -52,10 +55,6 @@ class TransactionTypeController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at | ||||
|      * * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/autocomplete/getTransactionTypesAC
 | ||||
|      */ | ||||
|     public function transactionTypes(AutocompleteRequest $request): JsonResponse | ||||
|     { | ||||
|         $data  = $request->getData(); | ||||
|   | ||||
| @@ -24,22 +24,19 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Controllers\Chart; | ||||
| 
 | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Chart\ChartRequest; | ||||
| use FireflyIII\Api\V1\Requests\Data\DateRequest; | ||||
| use FireflyIII\Enums\AccountTypeEnum; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Account; | ||||
| use FireflyIII\Models\Preference; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Support\Chart\ChartData; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use FireflyIII\Support\Http\Api\ApiSupport; | ||||
| use FireflyIII\Support\Http\Api\CleansChartData; | ||||
| use FireflyIII\Support\Http\Api\CollectsAccountsFromFilter; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Support\Facades\Log; | ||||
| 
 | ||||
| /** | ||||
|  * Class AccountController | ||||
| @@ -47,9 +44,12 @@ use Illuminate\Http\JsonResponse; | ||||
| class AccountController extends Controller | ||||
| { | ||||
|     use ApiSupport; | ||||
|     use CleansChartData; | ||||
|     use CollectsAccountsFromFilter; | ||||
| 
 | ||||
|     private ChartData                  $chartData; | ||||
|     protected array $acceptedRoles            = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     private array                  $chartData = []; | ||||
|     private AccountRepositoryInterface $repository; | ||||
| 
 | ||||
|     /** | ||||
| @@ -60,11 +60,10 @@ class AccountController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 /** @var User $user */ | ||||
|                 $user             = auth()->user(); | ||||
|                 $this->chartData  = new ChartData(); | ||||
|                 $this->repository = app(AccountRepositoryInterface::class); | ||||
|                 $this->repository->setUser($user); | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
|                 $this->repository->setUser($this->user); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -72,11 +71,9 @@ class AccountController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * TODO fix documentation | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function dashboard(ChartRequest $request): JsonResponse | ||||
|     public function overview(ChartRequest $request): JsonResponse | ||||
|     { | ||||
|         $queryParameters = $request->getParameters(); | ||||
|         $accounts        = $this->getAccountList($queryParameters); | ||||
| @@ -84,14 +81,16 @@ class AccountController extends Controller | ||||
|         // move date to end of day
 | ||||
|         $queryParameters['start']->startOfDay(); | ||||
|         $queryParameters['end']->endOfDay(); | ||||
|         // Log::debug(sprintf('dashboard(), convert to primary: %s', var_export($this->convertToPrimary, true)));
 | ||||
| 
 | ||||
|         // loop each account, and collect info:
 | ||||
|         /** @var Account $account */ | ||||
|         foreach ($accounts as $account) { | ||||
|             Log::debug(sprintf('Account #%d ("%s")', $account->id, $account->name)); | ||||
|             $this->renderAccountData($queryParameters, $account); | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($this->chartData->render()); | ||||
|         return response()->json($this->clean($this->chartData)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -99,109 +98,68 @@ class AccountController extends Controller | ||||
|      */ | ||||
|     private function renderAccountData(array $params, Account $account): void | ||||
|     { | ||||
|         $currency     = $this->repository->getAccountCurrency($account); | ||||
|         Log::debug(sprintf('Now in %s(array, #%d)', __METHOD__, $account->id)); | ||||
|         $currency          = $this->repository->getAccountCurrency($account); | ||||
|         $currentStart      = clone $params['start']; | ||||
|         $range             = Steam::finalAccountBalanceInRange($account, $params['start'], clone $params['end'], $this->convertToPrimary); | ||||
| 
 | ||||
| 
 | ||||
|         $previous          = array_values($range)[0]['balance']; | ||||
|         $pcPrevious        = null; | ||||
|         if (!$currency instanceof TransactionCurrency) { | ||||
|             $currency = $this->default; | ||||
|             $currency = $this->primaryCurrency; | ||||
|         } | ||||
|         $currentSet   = [ | ||||
|         $currentSet        = [ | ||||
|             'label'                   => $account->name, | ||||
| 
 | ||||
|             // the currency that belongs to the account.
 | ||||
|             'currency_id'             => (string) $currency->id, | ||||
|             'currency_id'             => (string)$currency->id, | ||||
|             'currency_name'           => $currency->name, | ||||
|             'currency_code'           => $currency->code, | ||||
|             'currency_symbol'         => $currency->symbol, | ||||
|             'currency_decimal_places' => $currency->decimal_places, | ||||
| 
 | ||||
|             // the primary currency
 | ||||
|             'primary_currency_id'     => (string)$this->primaryCurrency->id, | ||||
| 
 | ||||
|             // the default currency of the user (could be the same!)
 | ||||
|             'date'                    => $params['start']->toAtomString(), | ||||
|             'start'                   => $params['start']->toAtomString(), | ||||
|             'end'                     => $params['end']->toAtomString(), | ||||
|             'start_date'              => $params['start']->toAtomString(), | ||||
|             'end_date'                => $params['end']->toAtomString(), | ||||
|             'type'                    => 'line', | ||||
|             'yAxisID'                 => 0, | ||||
|             'period'                  => '1D', | ||||
|             'entries'                 => [], | ||||
|             'pc_entries'              => [], | ||||
|         ]; | ||||
|         $currentStart = clone $params['start']; | ||||
|         $range        = Steam::finalAccountBalanceInRange($account, $params['start'], clone $params['end'], $this->convertToNative); | ||||
|         if ($this->convertToPrimary) { | ||||
|             $currentSet['pc_entries']                      = []; | ||||
|             $currentSet['primary_currency_id']             = (string)$this->primaryCurrency->id; | ||||
|             $currentSet['primary_currency_code']           = $this->primaryCurrency->code; | ||||
|             $currentSet['primary_currency_symbol']         = $this->primaryCurrency->symbol; | ||||
|             $currentSet['primary_currency_decimal_places'] = $this->primaryCurrency->decimal_places; | ||||
|             $pcPrevious                                    = array_values($range)[0]['pc_balance']; | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|         $previous     = array_values($range)[0]['balance']; | ||||
|         while ($currentStart <= $params['end']) { | ||||
|             $format                        = $currentStart->format('Y-m-d'); | ||||
|             $label                         = $currentStart->toAtomString(); | ||||
|             $balance                       = array_key_exists($format, $range) ? $range[$format]['balance'] : $previous; | ||||
|             $previous                      = $balance; | ||||
|             $currentSet['entries'][$label] = $balance; | ||||
| 
 | ||||
| 
 | ||||
|             // do the same for the primary currency balance, if relevant:
 | ||||
|             $pcBalance                     = null; | ||||
|             if ($this->convertToPrimary) { | ||||
|                 $pcBalance                        = array_key_exists($format, $range) ? $range[$format]['pc_balance'] : $pcPrevious; | ||||
|                 $pcPrevious                       = $pcBalance; | ||||
|                 $currentSet['pc_entries'][$label] = $pcBalance; | ||||
|             } | ||||
| 
 | ||||
|             $currentStart->addDay(); | ||||
|             $currentSet['entries'][$label] = $balance; | ||||
|         } | ||||
|         $this->chartData->add($currentSet); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/charts/getChartAccountOverview
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function overview(DateRequest $request): JsonResponse | ||||
|     { | ||||
|         // parameters for chart:
 | ||||
|         $dates      = $request->getAll(); | ||||
| 
 | ||||
|         /** @var Carbon $start */ | ||||
|         $start      = $dates['start']; | ||||
| 
 | ||||
|         /** @var Carbon $end */ | ||||
|         $end        = $dates['end']; | ||||
| 
 | ||||
|         // set dates to end of day + start of day:
 | ||||
|         $start->startOfDay(); | ||||
|         $end->endOfDay(); | ||||
| 
 | ||||
|         // user's preferences
 | ||||
|         $defaultSet = $this->repository->getAccountsByType([AccountTypeEnum::ASSET->value])->pluck('id')->toArray(); | ||||
| 
 | ||||
|         /** @var Preference $frontpage */ | ||||
|         $frontpage  = app('preferences')->get('frontpageAccounts', $defaultSet); | ||||
| 
 | ||||
|         if (!(is_array($frontpage->data) && count($frontpage->data) > 0)) { | ||||
|             $frontpage->data = $defaultSet; | ||||
|             $frontpage->save(); | ||||
|         } | ||||
| 
 | ||||
|         // get accounts:
 | ||||
|         $accounts   = $this->repository->getAccountsById($frontpage->data); | ||||
|         $chartData  = []; | ||||
| 
 | ||||
|         /** @var Account $account */ | ||||
|         foreach ($accounts as $account) { | ||||
|             $currency     = $this->repository->getAccountCurrency($account) ?? $this->nativeCurrency; | ||||
|             $field        = $this->convertToNative && $currency->id !== $this->nativeCurrency->id ? 'native_balance' : 'balance'; | ||||
|             $currentSet   = [ | ||||
|                 'label'                   => $account->name, | ||||
|                 'currency_id'             => (string) $currency->id, | ||||
|                 'currency_code'           => $currency->code, | ||||
|                 'currency_symbol'         => $currency->symbol, | ||||
|                 'currency_decimal_places' => $currency->decimal_places, | ||||
|                 'start_date'              => $start->toAtomString(), | ||||
|                 'end_date'                => $end->toAtomString(), | ||||
|                 'type'                    => 'line', // line, area or bar
 | ||||
|                 'yAxisID'                 => 0, // 0, 1, 2
 | ||||
|                 'entries'                 => [], | ||||
|             ]; | ||||
|             // TODO this code is also present in the V2 chart account controller so this method is due to be deprecated.
 | ||||
|             $currentStart = clone $start; | ||||
|             $range        = Steam::finalAccountBalanceInRange($account, $start, clone $end, $this->convertToNative); | ||||
|             $previous     = array_values($range)[0][$field]; | ||||
|             while ($currentStart <= $end) { | ||||
|                 $format                        = $currentStart->format('Y-m-d'); | ||||
|                 $label                         = $currentStart->toAtomString(); | ||||
|                 $balance                       = array_key_exists($format, $range) ? $range[$format][$field] : $previous; | ||||
|                 $previous                      = $balance; | ||||
|                 $currentStart->addDay(); | ||||
|                 $currentSet['entries'][$label] = $balance; | ||||
|             } | ||||
|             $chartData[]  = $currentSet; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($chartData); | ||||
|         $this->chartData[] = $currentSet; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -1,8 +1,9 @@ | ||||
| <?php | ||||
| 
 | ||||
| 
 | ||||
| /* | ||||
|  * BalanceController.php | ||||
|  * Copyright (c) 2023 james@firefly-iii.org | ||||
|  * Copyright (c) 2025 james@firefly-iii.org | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -22,16 +23,15 @@ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Controllers\Chart; | ||||
| namespace FireflyIII\Api\V1\Controllers\Chart; | ||||
| 
 | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Api\V2\Request\Chart\ChartRequest; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Chart\ChartRequest; | ||||
| use FireflyIII\Enums\TransactionTypeEnum; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Support\Chart\ChartData; | ||||
| use FireflyIII\Support\Http\Api\AccountBalanceGrouped; | ||||
| use FireflyIII\Support\Http\Api\CleansChartData; | ||||
| use FireflyIII\Support\Http\Api\CollectsAccountsFromFilter; | ||||
| @@ -44,8 +44,9 @@ class BalanceController extends Controller | ||||
| { | ||||
|     use CleansChartData; | ||||
|     use CollectsAccountsFromFilter; | ||||
|     protected array $acceptedRoles            = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     private ChartData                  $chartData; | ||||
|     private array                  $chartData = []; | ||||
|     private GroupCollectorInterface    $collector; | ||||
|     private AccountRepositoryInterface $repository; | ||||
| 
 | ||||
| @@ -56,13 +57,13 @@ class BalanceController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(AccountRepositoryInterface::class); | ||||
|                 $this->collector  = app(GroupCollectorInterface::class); | ||||
|                 $userGroup        = $this->validateUserGroup($request); | ||||
|                 $this->repository->setUserGroup($userGroup); | ||||
|                 $this->collector->setUserGroup($userGroup); | ||||
|                 $this->chartData  = new ChartData(); | ||||
|                 // $this->default    = app('amount')->getNativeCurrency();
 | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
|                 $this->collector->setUserGroup($this->userGroup); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->collector->setUser($this->user); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -85,10 +86,6 @@ class BalanceController extends Controller | ||||
|         $queryParameters = $request->getParameters(); | ||||
|         $accounts        = $this->getAccountList($queryParameters); | ||||
| 
 | ||||
|         // prepare for currency conversion and data collection:
 | ||||
|         /** @var TransactionCurrency $default */ | ||||
|         $default         = app('amount')->getNativeCurrency(); | ||||
| 
 | ||||
|         // get journals for entire period:
 | ||||
| 
 | ||||
|         $this->collector->setRange($queryParameters['start'], $queryParameters['end']) | ||||
| @@ -100,7 +97,7 @@ class BalanceController extends Controller | ||||
| 
 | ||||
|         $object          = new AccountBalanceGrouped(); | ||||
|         $object->setPreferredRange($queryParameters['period']); | ||||
|         $object->setDefault($default); | ||||
|         $object->setPrimary($this->primaryCurrency); | ||||
|         $object->setAccounts($accounts); | ||||
|         $object->setJournals($journals); | ||||
|         $object->setStart($queryParameters['start']); | ||||
| @@ -108,9 +105,10 @@ class BalanceController extends Controller | ||||
|         $object->groupByCurrencyAndPeriod(); | ||||
|         $data            = $object->convertToChartData(); | ||||
|         foreach ($data as $entry) { | ||||
|             $this->chartData->add($entry); | ||||
|             $this->chartData[] = $entry; | ||||
|         } | ||||
|         $this->chartData = $this->clean($this->chartData); | ||||
| 
 | ||||
|         return response()->json($this->chartData->render()); | ||||
|         return response()->json($this->chartData); | ||||
|     } | ||||
| } | ||||
| @@ -26,16 +26,17 @@ namespace FireflyIII\Api\V1\Controllers\Chart; | ||||
| 
 | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Generic\DateRequest; | ||||
| use FireflyIII\Api\V1\Requests\Data\SameDateRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Budget; | ||||
| use FireflyIII\Models\BudgetLimit; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\Budget\BudgetLimitRepositoryInterface; | ||||
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | ||||
| use FireflyIII\Repositories\Budget\OperationsRepositoryInterface; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Http\Api\CleansChartData; | ||||
| use FireflyIII\Support\Http\Api\ExchangeRateConverter; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Support\Collection; | ||||
| @@ -54,7 +55,6 @@ class BudgetController extends Controller | ||||
|     protected OperationsRepositoryInterface $opsRepository; | ||||
|     private BudgetLimitRepositoryInterface  $blRepository; | ||||
|     private array                           $currencies = []; | ||||
|     private TransactionCurrency             $currency; | ||||
|     private BudgetRepositoryInterface       $repository; | ||||
| 
 | ||||
|     public function __construct() | ||||
| @@ -62,13 +62,16 @@ class BudgetController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository    = app(BudgetRepositoryInterface::class); | ||||
|                 $this->blRepository  = app(BudgetLimitRepositoryInterface::class); | ||||
|                 $this->opsRepository = app(OperationsRepositoryInterface::class); | ||||
|                 $userGroup           = $this->validateUserGroup($request); | ||||
|                 $this->repository->setUserGroup($userGroup); | ||||
|                 $this->opsRepository->setUserGroup($userGroup); | ||||
|                 $this->blRepository->setUserGroup($userGroup); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
|                 $this->opsRepository->setUserGroup($this->userGroup); | ||||
|                 $this->blRepository->setUserGroup($this->userGroup); | ||||
|                 $this->repository->setUser($this->user); | ||||
|                 $this->opsRepository->setUser($this->user); | ||||
|                 $this->blRepository->setUser($this->user); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -77,8 +80,10 @@ class BudgetController extends Controller | ||||
| 
 | ||||
|     /** | ||||
|      * TODO see autocomplete/accountcontroller | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function dashboard(DateRequest $request): JsonResponse | ||||
|     public function overview(SameDateRequest $request): JsonResponse | ||||
|     { | ||||
|         $params  = $request->getAll(); | ||||
| 
 | ||||
| @@ -107,34 +112,87 @@ class BudgetController extends Controller | ||||
|     private function processBudget(Budget $budget, Carbon $start, Carbon $end): array | ||||
|     { | ||||
|         // get all limits:
 | ||||
|         $limits = $this->blRepository->getBudgetLimits($budget, $start, $end); | ||||
|         $rows   = []; | ||||
|         $limits     = $this->blRepository->getBudgetLimits($budget, $start, $end); | ||||
|         $rows       = []; | ||||
|         $spent      = $this->opsRepository->listExpenses($start, $end, null, new Collection()->push($budget)); | ||||
|         $expenses   = $this->processExpenses($budget->id, $spent, $start, $end); | ||||
|         $converter  = new ExchangeRateConverter(); | ||||
|         $currencies = [$this->primaryCurrency->id => $this->primaryCurrency]; | ||||
| 
 | ||||
|         // if no limits
 | ||||
|         if (0 === $limits->count()) { | ||||
|             // return as a single item in an array
 | ||||
|             $rows = $this->noBudgetLimits($budget, $start, $end); | ||||
|         } | ||||
|         if ($limits->count() > 0) { | ||||
|             $rows = $this->budgetLimits($budget, $limits); | ||||
|         /** | ||||
|          * @var int   $currencyId | ||||
|          * @var array $row | ||||
|          */ | ||||
|         foreach ($expenses as $currencyId => $row) { | ||||
|             // budgeted, left and overspent are now 0.
 | ||||
|             $limit               = $this->filterLimit($currencyId, $limits); | ||||
| 
 | ||||
|             // primary currency entries
 | ||||
|             $row['pc_budgeted']  = '0'; | ||||
|             $row['pc_spent']     = '0'; | ||||
|             $row['pc_left']      = '0'; | ||||
|             $row['pc_overspent'] = '0'; | ||||
| 
 | ||||
|             if ($limit instanceof BudgetLimit) { | ||||
|                 $row['budgeted']  = $limit->amount; | ||||
|                 $row['left']      = bcsub((string) $row['budgeted'], bcmul((string) $row['spent'], '-1')); | ||||
|                 $row['overspent'] = bcmul($row['left'], '-1'); | ||||
|                 $row['left']      = 1 === bccomp($row['left'], '0') ? $row['left'] : '0'; | ||||
|                 $row['overspent'] = 1 === bccomp($row['overspent'], '0') ? $row['overspent'] : '0'; | ||||
|             } | ||||
| 
 | ||||
|             // convert data if necessary.
 | ||||
|             if (true === $this->convertToPrimary && $currencyId !== $this->primaryCurrency->id) { | ||||
|                 $currencies[$currencyId] ??= Amount::getTransactionCurrencyById($currencyId); | ||||
|                 $row['pc_budgeted']  = $converter->convert($currencies[$currencyId], $this->primaryCurrency, $start, $row['budgeted']); | ||||
|                 $row['pc_spent']     = $converter->convert($currencies[$currencyId], $this->primaryCurrency, $start, $row['spent']); | ||||
|                 $row['pc_left']      = $converter->convert($currencies[$currencyId], $this->primaryCurrency, $start, $row['left']); | ||||
|                 $row['pc_overspent'] = $converter->convert($currencies[$currencyId], $this->primaryCurrency, $start, $row['overspent']); | ||||
|             } | ||||
|             if (true === $this->convertToPrimary && $currencyId === $this->primaryCurrency->id) { | ||||
|                 $row['pc_budgeted']  = $row['budgeted']; | ||||
|                 $row['pc_spent']     = $row['spent']; | ||||
|                 $row['pc_left']      = $row['left']; | ||||
|                 $row['pc_overspent'] = $row['overspent']; | ||||
|             } | ||||
|             $rows[]              = $row; | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
|         // is always an array
 | ||||
|         $return = []; | ||||
|         $return     = []; | ||||
|         foreach ($rows as $row) { | ||||
|             $current  = [ | ||||
|                 'label'                   => $budget->name, | ||||
|                 'currency_id'             => (string) $row['currency_id'], | ||||
|                 'currency_code'           => $row['currency_code'], | ||||
|                 'currency_name'           => $row['currency_name'], | ||||
|                 'currency_decimal_places' => $row['currency_decimal_places'], | ||||
|                 'period'                  => null, | ||||
|                 'start'                   => $row['start'], | ||||
|                 'end'                     => $row['end'], | ||||
|                 'entries'                 => [ | ||||
|                 'label'                           => $budget->name, | ||||
|                 'currency_id'                     => (string)$row['currency_id'], | ||||
|                 'currency_name'                   => $row['currency_name'], | ||||
|                 'currency_code'                   => $row['currency_code'], | ||||
|                 'currency_decimal_places'         => $row['currency_decimal_places'], | ||||
| 
 | ||||
|                 'primary_currency_id'             => (string)$this->primaryCurrency->id, | ||||
|                 'primary_currency_name'           => $this->primaryCurrency->name, | ||||
|                 'primary_currency_code'           => $this->primaryCurrency->code, | ||||
|                 'primary_currency_symbol'         => $this->primaryCurrency->symbol, | ||||
|                 'primary_currency_decimal_places' => $this->primaryCurrency->decimal_places, | ||||
| 
 | ||||
|                 'period'                          => null, | ||||
|                 'date'                            => $row['start'], | ||||
|                 'start_date'                      => $row['start'], | ||||
|                 'end_date'                        => $row['end'], | ||||
|                 'yAxisID'                         => 0, | ||||
|                 'type'                            => 'bar', | ||||
|                 'entries'                         => [ | ||||
|                     'budgeted'  => $row['budgeted'], | ||||
|                     'spent'     => $row['spent'], | ||||
|                     'left'      => $row['left'], | ||||
|                     'overspent' => $row['overspent'], | ||||
|                 ], | ||||
|                 'pc_entries'                      => [ | ||||
|                     'budgeted'  => $row['pc_budgeted'], | ||||
|                     'spent'     => $row['pc_spent'], | ||||
|                     'left'      => $row['pc_left'], | ||||
|                     'overspent' => $row['pc_overspent'], | ||||
|                 ], | ||||
|             ]; | ||||
|             $return[] = $current; | ||||
|         } | ||||
| @@ -142,28 +200,26 @@ class BudgetController extends Controller | ||||
|         return $return; | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * When no budget limits are present, the expenses of the whole period are collected and grouped. | ||||
|      * This is grouped per currency. Because there is no limit set, "left to spend" and "overspent" are empty. | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     private function noBudgetLimits(Budget $budget, Carbon $start, Carbon $end): array | ||||
|     { | ||||
|         $spent = $this->opsRepository->listExpenses($start, $end, null, new Collection([$budget])); | ||||
| 
 | ||||
|         return $this->processExpenses($budget->id, $spent, $start, $end); | ||||
|     } | ||||
|     //    /**
 | ||||
|     //     * When no budget limits are present, the expenses of the whole period are collected and grouped.
 | ||||
|     //     * This is grouped per currency. Because there is no limit set, "left to spend" and "overspent" are empty.
 | ||||
|     //     *
 | ||||
|     //     * @throws FireflyException
 | ||||
|     //     */
 | ||||
|     //    private function noBudgetLimits(Budget $budget, Carbon $start, Carbon $end): array
 | ||||
|     //    {
 | ||||
|     //        $spent = $this->opsRepository->listExpenses($start, $end, null, new Collection()->push($budget));
 | ||||
|     //
 | ||||
|     //        return $this->processExpenses($budget->id, $spent, $start, $end);
 | ||||
|     //    }
 | ||||
| 
 | ||||
|     /** | ||||
|      * Shared between the "noBudgetLimits" function and "processLimit". Will take a single set of expenses and return | ||||
|      * its info. | ||||
|      * | ||||
|      * @param array<int, array<int, string>> $array | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     private function processExpenses(int $budgetId, array $array, Carbon $start, Carbon $end): array | ||||
|     private function processExpenses(int $budgetId, array $spent, Carbon $start, Carbon $end): array | ||||
|     { | ||||
|         $return = []; | ||||
| 
 | ||||
| @@ -174,16 +230,17 @@ class BudgetController extends Controller | ||||
|          * @var int   $currencyId | ||||
|          * @var array $block | ||||
|          */ | ||||
|         foreach ($array as $currencyId => $block) { | ||||
|             $this->currencies[$currencyId] ??= TransactionCurrency::find($currencyId); | ||||
|         foreach ($spent as $currencyId => $block) { | ||||
|             $this->currencies[$currencyId] ??= Amount::getTransactionCurrencyById($currencyId); | ||||
|             $return[$currencyId]           ??= [ | ||||
|                 'currency_id'             => (string) $currencyId, | ||||
|                 'currency_id'             => (string)$currencyId, | ||||
|                 'currency_code'           => $block['currency_code'], | ||||
|                 'currency_name'           => $block['currency_name'], | ||||
|                 'currency_symbol'         => $block['currency_symbol'], | ||||
|                 'currency_decimal_places' => (int) $block['currency_decimal_places'], | ||||
|                 'currency_decimal_places' => (int)$block['currency_decimal_places'], | ||||
|                 'start'                   => $start->toAtomString(), | ||||
|                 'end'                     => $end->toAtomString(), | ||||
|                 'budgeted'                => '0', | ||||
|                 'spent'                   => '0', | ||||
|                 'left'                    => '0', | ||||
|                 'overspent'               => '0', | ||||
| @@ -193,68 +250,101 @@ class BudgetController extends Controller | ||||
|             // var_dump($return);
 | ||||
|             /** @var array $journal */ | ||||
|             foreach ($currentBudgetArray['transaction_journals'] as $journal) { | ||||
|                 $return[$currencyId]['spent'] = bcadd($return[$currencyId]['spent'], (string) $journal['amount']); | ||||
|                 /** @var numeric-string $amount */ | ||||
|                 $amount                       = (string)$journal['amount']; | ||||
|                 $return[$currencyId]['spent'] = bcadd($return[$currencyId]['spent'], $amount); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         return $return; | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Function that processes each budget limit (per budget). | ||||
|      * | ||||
|      * If you have a budget limit in EUR, only transactions in EUR will be considered. | ||||
|      * If you have a budget limit in GBP, only transactions in GBP will be considered. | ||||
|      * | ||||
|      * If you have a budget limit in EUR, and a transaction in GBP, it will not be considered for the EUR budget limit. | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     private function budgetLimits(Budget $budget, Collection $limits): array | ||||
|     //    /**
 | ||||
|     //     * Function that processes each budget limit (per budget).
 | ||||
|     //     *
 | ||||
|     //     * If you have a budget limit in EUR, only transactions in EUR will be considered.
 | ||||
|     //     * If you have a budget limit in GBP, only transactions in GBP will be considered.
 | ||||
|     //     *
 | ||||
|     //     * If you have a budget limit in EUR, and a transaction in GBP, it will not be considered for the EUR budget limit.
 | ||||
|     //     *
 | ||||
|     //     * @throws FireflyException
 | ||||
|     //     */
 | ||||
|     //    private function budgetLimits(Budget $budget, Collection $limits): array
 | ||||
|     //    {
 | ||||
|     //        Log::debug(sprintf('Now in budgetLimits(#%d)', $budget->id));
 | ||||
|     //        $data = [];
 | ||||
|     //
 | ||||
|     //        /** @var BudgetLimit $limit */
 | ||||
|     //        foreach ($limits as $limit) {
 | ||||
|     //            $data = array_merge($data, $this->processLimit($budget, $limit));
 | ||||
|     //        }
 | ||||
|     //
 | ||||
|     //        return $data;
 | ||||
|     //    }
 | ||||
| 
 | ||||
|     //    /**
 | ||||
|     //     * @throws FireflyException
 | ||||
|     //     */
 | ||||
|     //    private function processLimit(Budget $budget, BudgetLimit $limit): array
 | ||||
|     //    {
 | ||||
|     //        Log::debug(sprintf('Created new ExchangeRateConverter in %s', __METHOD__));
 | ||||
|     //        $end             = clone $limit->end_date;
 | ||||
|     //        $end->endOfDay();
 | ||||
|     //        $spent           = $this->opsRepository->listExpenses($limit->start_date, $end, null, new Collection()->push($budget));
 | ||||
|     //        $limitCurrencyId = $limit->transaction_currency_id;
 | ||||
|     //
 | ||||
|     //        /** @var array $entry */
 | ||||
|     //        // only spent the entry where the entry's currency matches the budget limit's currency
 | ||||
|     //        // so $filtered will only have 1 or 0 entries
 | ||||
|     //        $filtered        = array_filter($spent, fn ($entry) => $entry['currency_id'] === $limitCurrencyId);
 | ||||
|     //        $result          = $this->processExpenses($budget->id, $filtered, $limit->start_date, $end);
 | ||||
|     //        if (1 === count($result)) {
 | ||||
|     //            $compare                              = bccomp($limit->amount, (string)app('steam')->positive($result[$limitCurrencyId]['spent']));
 | ||||
|     //            $result[$limitCurrencyId]['budgeted'] = $limit->amount;
 | ||||
|     //            if (1 === $compare) {
 | ||||
|     //                // convert this amount into the primary currency:
 | ||||
|     //                $result[$limitCurrencyId]['left'] = bcadd($limit->amount, (string)$result[$limitCurrencyId]['spent']);
 | ||||
|     //            }
 | ||||
|     //            if ($compare <= 0) {
 | ||||
|     //                $result[$limitCurrencyId]['overspent'] = app('steam')->positive(bcadd($limit->amount, (string)$result[$limitCurrencyId]['spent']));
 | ||||
|     //            }
 | ||||
|     //        }
 | ||||
|     //
 | ||||
|     //        return $result;
 | ||||
|     //    }
 | ||||
| 
 | ||||
|     private function filterLimit(int $currencyId, Collection $limits): ?BudgetLimit | ||||
|     { | ||||
|         Log::debug(sprintf('Now in budgetLimits(#%d)', $budget->id)); | ||||
|         $data = []; | ||||
|         $amount    = '0'; | ||||
|         $limit     = null; | ||||
|         $converter = new ExchangeRateConverter(); | ||||
| 
 | ||||
|         /** @var BudgetLimit $limit */ | ||||
|         foreach ($limits as $limit) { | ||||
|             $data = array_merge($data, $this->processLimit($budget, $limit)); | ||||
|         } | ||||
| 
 | ||||
|         return $data; | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     private function processLimit(Budget $budget, BudgetLimit $limit): array | ||||
|     { | ||||
|         Log::debug(sprintf('Created new ExchangeRateConverter in %s', __METHOD__)); | ||||
|         $end             = clone $limit->end_date; | ||||
|         $end->endOfDay(); | ||||
|         $spent           = $this->opsRepository->listExpenses($limit->start_date, $end, null, new Collection([$budget])); | ||||
|         $limitCurrencyId = $limit->transaction_currency_id; | ||||
|         $filtered        = []; | ||||
| 
 | ||||
|         /** @var array $entry */ | ||||
|         foreach ($spent as $currencyId => $entry) { | ||||
|             // only spent the entry where the entry's currency matches the budget limit's currency
 | ||||
|             // so $filtered will only have 1 or 0 entries
 | ||||
|             if ($entry['currency_id'] === $limitCurrencyId) { | ||||
|                 $filtered[$currencyId] = $entry; | ||||
|         /** @var BudgetLimit $current */ | ||||
|         foreach ($limits as $current) { | ||||
|             if (true === $this->convertToPrimary) { | ||||
|                 if ($current->transaction_currency_id === $this->primaryCurrency->id) { | ||||
|                     // simply add it.
 | ||||
|                     $amount = bcadd($amount, (string)$current->amount); | ||||
|                     Log::debug(sprintf('Set amount in limit to %s', $amount)); | ||||
|                 } | ||||
|                 if ($current->transaction_currency_id !== $this->primaryCurrency->id) { | ||||
|                     // convert and then add it.
 | ||||
|                     $converted = $converter->convert($current->transactionCurrency, $this->primaryCurrency, $current->start_date, $current->amount); | ||||
|                     $amount    = bcadd($amount, $converted); | ||||
|                     Log::debug(sprintf('Budgeted in limit #%d: %s %s, converted to %s %s', $current->id, $current->transactionCurrency->code, $current->amount, $this->primaryCurrency->code, $converted)); | ||||
|                     Log::debug(sprintf('Set amount in limit to %s', $amount)); | ||||
|                 } | ||||
|             } | ||||
|             if ($current->transaction_currency_id === $currencyId) { | ||||
|                 $limit = $current; | ||||
|             } | ||||
|         } | ||||
|         $result          = $this->processExpenses($budget->id, $filtered, $limit->start_date, $end); | ||||
|         if (1 === count($result)) { | ||||
|             $compare = bccomp($limit->amount, (string) app('steam')->positive($result[$limitCurrencyId]['spent'])); | ||||
|             if (1 === $compare) { | ||||
|                 // convert this amount into the native currency:
 | ||||
|                 $result[$limitCurrencyId]['left'] = bcadd($limit->amount, (string) $result[$limitCurrencyId]['spent']); | ||||
|             } | ||||
|             if ($compare <= 0) { | ||||
|                 $result[$limitCurrencyId]['overspent'] = app('steam')->positive(bcadd($limit->amount, (string) $result[$limitCurrencyId]['spent'])); | ||||
|             } | ||||
|         if (null !== $limit && true === $this->convertToPrimary) { | ||||
|             // convert and add all amounts.
 | ||||
|             $limit->amount = app('steam')->positive($amount); | ||||
|             Log::debug(sprintf('Final amount in limit with converted amount %s', $limit->amount)); | ||||
|         } | ||||
| 
 | ||||
|         return $result; | ||||
|         return $limit; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -25,17 +25,21 @@ declare(strict_types=1); | ||||
| namespace FireflyIII\Api\V1\Controllers\Chart; | ||||
| 
 | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Api\V2\Request\Generic\DateRequest; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Data\SameDateRequest; | ||||
| use FireflyIII\Enums\AccountTypeEnum; | ||||
| use FireflyIII\Enums\TransactionTypeEnum; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Repositories\Currency\CurrencyRepositoryInterface; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use FireflyIII\Support\Http\Api\CleansChartData; | ||||
| use FireflyIII\Support\Http\Api\ExchangeRateConverter; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Support\Facades\Log; | ||||
| 
 | ||||
| /** | ||||
|  * Class BudgetController | ||||
| @@ -45,6 +49,8 @@ class CategoryController extends Controller | ||||
|     use CleansChartData; | ||||
|     use ValidatesUserGroupTrait; | ||||
| 
 | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     private AccountRepositoryInterface  $accountRepos; | ||||
|     private CurrencyRepositoryInterface $currencyRepos; | ||||
| 
 | ||||
| @@ -53,11 +59,13 @@ class CategoryController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->accountRepos  = app(AccountRepositoryInterface::class); | ||||
|                 $this->currencyRepos = app(CurrencyRepositoryInterface::class); | ||||
|                 $userGroup           = $this->validateUserGroup($request); | ||||
|                 $this->accountRepos->setUserGroup($userGroup); | ||||
|                 $this->currencyRepos->setUserGroup($userGroup); | ||||
|                 $this->accountRepos->setUserGroup($this->userGroup); | ||||
|                 $this->currencyRepos->setUserGroup($this->userGroup); | ||||
|                 $this->accountRepos->setUser($this->user); | ||||
|                 $this->currencyRepos->setUser($this->user); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -72,54 +80,108 @@ class CategoryController extends Controller | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
|      */ | ||||
|     public function dashboard(DateRequest $request): JsonResponse | ||||
|     public function overview(SameDateRequest $request): JsonResponse | ||||
|     { | ||||
|         /** @var Carbon $start */ | ||||
|         $start      = $this->parameters->get('start'); | ||||
| 
 | ||||
|         /** @var Carbon $end */ | ||||
|         $end        = $this->parameters->get('end'); | ||||
|         $accounts   = $this->accountRepos->getAccountsByType([AccountTypeEnum::DEBT->value, AccountTypeEnum::LOAN->value, AccountTypeEnum::MORTGAGE->value, AccountTypeEnum::ASSET->value, AccountTypeEnum::DEFAULT->value]); | ||||
|         $accounts   = $this->accountRepos->getAccountsByType([AccountTypeEnum::DEBT->value, AccountTypeEnum::LOAN->value, AccountTypeEnum::MORTGAGE->value, AccountTypeEnum::ASSET->value]); | ||||
|         $currencies = []; | ||||
|         $return     = []; | ||||
|         $converter  = new ExchangeRateConverter(); | ||||
| 
 | ||||
|         // get journals for entire period:
 | ||||
|         /** @var GroupCollectorInterface $collector */ | ||||
|         $collector  = app(GroupCollectorInterface::class); | ||||
|         $collector->setRange($start, $end)->withAccountInformation(); | ||||
|         $collector->setXorAccounts($accounts)->withCategoryInformation(); | ||||
|         $collector->setTypes([TransactionTypeEnum::WITHDRAWAL->value, TransactionTypeEnum::RECONCILIATION->value]); | ||||
|         $collector->setTypes([TransactionTypeEnum::WITHDRAWAL->value, TransactionTypeEnum::DEPOSIT->value]); | ||||
|         $journals   = $collector->getExtractedJournals(); | ||||
| 
 | ||||
|         /** @var array $journal */ | ||||
|         foreach ($journals as $journal) { | ||||
|             $currencyId              = (int) $journal['currency_id']; | ||||
|             $currency                = $currencies[$currencyId] ?? $this->currencyRepos->find($currencyId); | ||||
|             $currencies[$currencyId] = $currency; | ||||
|             $categoryName            = $journal['category_name'] ?? (string) trans('firefly.no_category'); | ||||
|             $amount                  = app('steam')->positive($journal['amount']); | ||||
|             $key                     = sprintf('%s-%s', $categoryName, $currency->code); | ||||
|             // find journal:
 | ||||
|             $journalCurrencyId              = (int)$journal['currency_id']; | ||||
|             $type                           = $journal['transaction_type_type']; | ||||
|             $currency                       = $currencies[$journalCurrencyId] ?? $this->currencyRepos->find($journalCurrencyId); | ||||
|             $currencies[$journalCurrencyId] = $currency; | ||||
|             $currencyId                     = (int)$currency->id; | ||||
|             $currencyName                   = (string)$currency->name; | ||||
|             $currencyCode                   = (string)$currency->code; | ||||
|             $currencySymbol                 = (string)$currency->symbol; | ||||
|             $currencyDecimalPlaces          = (int)$currency->decimal_places; | ||||
|             $amount                         = Steam::positive((string)$journal['amount']); | ||||
|             $pcAmount                       = null; | ||||
| 
 | ||||
|             // overrule if necessary:
 | ||||
|             if ($this->convertToPrimary && $journalCurrencyId === $this->primaryCurrency->id) { | ||||
|                 $pcAmount = $amount; | ||||
|             } | ||||
|             if ($this->convertToPrimary && $journalCurrencyId !== $this->primaryCurrency->id) { | ||||
|                 $currencyId            = (int)$this->primaryCurrency->id; | ||||
|                 $currencyName          = (string)$this->primaryCurrency->name; | ||||
|                 $currencyCode          = (string)$this->primaryCurrency->code; | ||||
|                 $currencySymbol        = (string)$this->primaryCurrency->symbol; | ||||
|                 $currencyDecimalPlaces = (int)$this->primaryCurrency->decimal_places; | ||||
|                 $pcAmount              = $converter->convert($currency, $this->primaryCurrency, $journal['date'], $amount); | ||||
|                 Log::debug(sprintf('Converted %s %s to %s %s', $journal['currency_code'], $amount, $this->primaryCurrency->code, $pcAmount)); | ||||
|             } | ||||
| 
 | ||||
| 
 | ||||
|             $categoryName                   = $journal['category_name'] ?? (string)trans('firefly.no_category'); | ||||
|             $key                            = sprintf('%s-%s', $categoryName, $currencyCode); | ||||
|             // create arrays
 | ||||
|             $return[$key] ??= [ | ||||
|                 'label'                   => $categoryName, | ||||
|                 'currency_id'             => (string) $currency->id, | ||||
|                 'currency_code'           => $currency->code, | ||||
|                 'currency_name'           => $currency->name, | ||||
|                 'currency_symbol'         => $currency->symbol, | ||||
|                 'currency_decimal_places' => $currency->decimal_places, | ||||
|                 'period'                  => null, | ||||
|                 'start'                   => $start->toAtomString(), | ||||
|                 'end'                     => $end->toAtomString(), | ||||
|                 'amount'                  => '0', | ||||
|                 'label'                           => $categoryName, | ||||
|                 'currency_id'                     => (string)$currencyId, | ||||
|                 'currency_name'                   => $currencyName, | ||||
|                 'currency_code'                   => $currencyCode, | ||||
|                 'currency_symbol'                 => $currencySymbol, | ||||
|                 'currency_decimal_places'         => $currencyDecimalPlaces, | ||||
|                 'primary_currency_id'             => (string)$this->primaryCurrency->id, | ||||
|                 'primary_currency_name'           => (string)$this->primaryCurrency->name, | ||||
|                 'primary_currency_code'           => (string)$this->primaryCurrency->code, | ||||
|                 'primary_currency_symbol'         => (string)$this->primaryCurrency->symbol, | ||||
|                 'primary_currency_decimal_places' => (int)$this->primaryCurrency->decimal_places, | ||||
|                 'period'                          => null, | ||||
|                 'start_date'                      => $start->toAtomString(), | ||||
|                 'end_date'                        => $end->toAtomString(), | ||||
|                 'yAxisID'                         => 0, | ||||
|                 'type'                            => 'bar', | ||||
|                 'entries'                         => [ | ||||
|                     'spent'  => '0', | ||||
|                     'earned' => '0', | ||||
|                 ], | ||||
|                 'pc_entries'                      => [ | ||||
|                     'spent'  => '0', | ||||
|                     'earned' => '0', | ||||
|                 ], | ||||
|             ]; | ||||
| 
 | ||||
|             // add monies
 | ||||
|             $return[$key]['amount']  = bcadd($return[$key]['amount'], (string) $amount); | ||||
|             // expenses to spent
 | ||||
|             if (TransactionTypeEnum::WITHDRAWAL->value === $type) { | ||||
|                 $return[$key]['entries']['spent'] = bcadd($return[$key]['entries']['spent'], $amount); | ||||
|                 if (null !== $pcAmount) { | ||||
|                     $return[$key]['pc_entries']['spent'] = bcadd($return[$key]['pc_entries']['spent'], $pcAmount); | ||||
|                 } | ||||
| 
 | ||||
|                 continue; | ||||
|             } | ||||
|             // positive amount = earned
 | ||||
|             if (TransactionTypeEnum::DEPOSIT->value === $type) { | ||||
|                 $return[$key]['entries']['earned'] = bcadd($return[$key]['entries']['earned'], $amount); | ||||
|                 if (null !== $pcAmount) { | ||||
|                     $return[$key]['pc_entries']['earned'] = bcadd($return[$key]['pc_entries']['earned'], $pcAmount); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         $return     = array_values($return); | ||||
| 
 | ||||
|         // order by amount
 | ||||
|         usort($return, static fn (array $a, array $b) => (float) $a['amount'] < (float) $b['amount'] ? 1 : -1); | ||||
|         usort($return, static fn (array $a, array $b) => ((float)$a['entries']['spent'] + (float)$a['entries']['earned']) < ((float)$b['entries']['spent'] + (float)$b['entries']['earned']) ? 1 : -1); | ||||
| 
 | ||||
|         return response()->json($this->clean($return)); | ||||
|     } | ||||
|   | ||||
| @@ -27,7 +27,6 @@ namespace FireflyIII\Api\V1\Controllers; | ||||
| use Carbon\Carbon; | ||||
| use Carbon\Exceptions\InvalidFormatException; | ||||
| use FireflyIII\Exceptions\BadHttpHeaderException; | ||||
| use FireflyIII\Models\Preference; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| @@ -62,14 +61,12 @@ abstract class Controller extends BaseController | ||||
|     use ValidatesRequests; | ||||
|     use ValidatesUserGroupTrait; | ||||
| 
 | ||||
|     protected const string CONTENT_TYPE            = 'application/vnd.api+json'; | ||||
|     protected const string JSON_CONTENT_TYPE       = 'application/json'; | ||||
|     protected array               $accepts         = ['application/json', 'application/vnd.api+json']; | ||||
|     protected const string CONTENT_TYPE             = 'application/vnd.api+json'; | ||||
|     protected const string JSON_CONTENT_TYPE        = 'application/json'; | ||||
|     protected array $accepts                        = ['application/json', 'application/vnd.api+json']; | ||||
| 
 | ||||
|     /** @var array<int, string> */ | ||||
|     protected array               $allowedSort; | ||||
|     protected bool                $convertToNative = false; | ||||
|     protected TransactionCurrency $nativeCurrency; | ||||
|     protected bool                $convertToPrimary = false; | ||||
|     protected TransactionCurrency $primaryCurrency; | ||||
|     protected ParameterBag        $parameters; | ||||
| 
 | ||||
|     /** | ||||
| @@ -78,14 +75,13 @@ abstract class Controller extends BaseController | ||||
|     public function __construct() | ||||
|     { | ||||
|         // get global parameters
 | ||||
|         $this->allowedSort = config('firefly.allowed_sort_parameters'); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->parameters = $this->getParameters(); | ||||
|                 if (auth()->check()) { | ||||
|                     $language              = Steam::getLanguage(); | ||||
|                     $this->convertToNative = Amount::convertToNative(); | ||||
|                     $this->nativeCurrency  = Amount::getNativeCurrency(); | ||||
|                     $language               = Steam::getLanguage(); | ||||
|                     $this->convertToPrimary = Amount::convertToPrimary(); | ||||
|                     $this->primaryCurrency  = Amount::getPrimaryCurrency(); | ||||
|                     app()->setLocale($language); | ||||
|                 } | ||||
| 
 | ||||
| @@ -107,13 +103,8 @@ abstract class Controller extends BaseController | ||||
|     private function getParameters(): ParameterBag | ||||
|     { | ||||
|         $bag      = new ParameterBag(); | ||||
|         $page     = (int) request()->get('page'); | ||||
|         if ($page < 1) { | ||||
|             $page = 1; | ||||
|         } | ||||
|         if ($page > 2 ** 16) { | ||||
|             $page = 2 ** 16; | ||||
|         } | ||||
|         $page     = (int)request()->get('page'); | ||||
|         $page     = min(max(1, $page), 2 ** 16); | ||||
|         $bag->set('page', $page); | ||||
| 
 | ||||
|         // some date fields:
 | ||||
| @@ -127,24 +118,19 @@ abstract class Controller extends BaseController | ||||
|                 Log::error(sprintf('Request field "%s" contains a non-scalar value. Value set to NULL.', $field)); | ||||
|                 Log::error($e->getMessage()); | ||||
|                 Log::error($e->getTraceAsString()); | ||||
|                 $value = null; | ||||
|             } | ||||
|             $obj  = null; | ||||
|             if (null !== $date) { | ||||
|                 try { | ||||
|                     $obj = Carbon::parse((string) $date); | ||||
|                     $obj = Carbon::parse((string)$date, config('app.timezone')); | ||||
|                 } catch (InvalidFormatException $e) { | ||||
|                     // don't care
 | ||||
|                     Log::warning( | ||||
|                         sprintf( | ||||
|                             'Ignored invalid date "%s" in API controller parameter check: %s', | ||||
|                             substr((string) $date, 0, 20), | ||||
|                             $e->getMessage() | ||||
|                         ) | ||||
|                     ); | ||||
|                     Log::warning(sprintf('Ignored invalid date "%s" in API controller parameter check: %s', substr((string)$date, 0, 20), $e->getMessage())); | ||||
|                 } | ||||
|             } | ||||
|             $bag->set($field, $obj); | ||||
|             if ($obj instanceof Carbon) { | ||||
|                 $bag->set($field, $obj); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         // integer fields:
 | ||||
| @@ -159,14 +145,8 @@ abstract class Controller extends BaseController | ||||
|                 $value = null; | ||||
|             } | ||||
|             if (null !== $value) { | ||||
|                 $value = (int) $value; | ||||
|                 if ($value < 1) { | ||||
|                     $value = 1; | ||||
|                 } | ||||
|                 if ($value > 2 ** 16) { | ||||
|                     $value = 2 ** 16; | ||||
|                 } | ||||
| 
 | ||||
|                 $value = (int)$value; | ||||
|                 $value = min(max(1, $value), 2 ** 16); | ||||
|                 $bag->set($integer, $value); | ||||
|             } | ||||
|             if (null === $value | ||||
| @@ -176,46 +156,14 @@ abstract class Controller extends BaseController | ||||
|                 /** @var User $user */ | ||||
|                 $user     = auth()->user(); | ||||
| 
 | ||||
|                 /** @var Preference $pageSize */ | ||||
|                 $pageSize = (int) app('preferences')->getForUser($user, 'listPageSize', 50)->data; | ||||
|                 $pageSize = (int)app('preferences')->getForUser($user, 'listPageSize', 50)->data; | ||||
|                 $bag->set($integer, $pageSize); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         // sort fields:
 | ||||
|         return $this->getSortParameters($bag); | ||||
|     } | ||||
| 
 | ||||
|     private function getSortParameters(ParameterBag $bag): ParameterBag | ||||
|     { | ||||
|         $sortParameters = []; | ||||
| 
 | ||||
|         try { | ||||
|             $param = (string) request()->query->get('sort'); | ||||
|         } catch (BadRequestException $e) { | ||||
|             Log::error('Request field "sort" contains a non-scalar value. Value set to NULL.'); | ||||
|             Log::error($e->getMessage()); | ||||
|             Log::error($e->getTraceAsString()); | ||||
|             $param = ''; | ||||
|         } | ||||
|         if ('' === $param) { | ||||
|             return $bag; | ||||
|         } | ||||
|         $parts          = explode(',', $param); | ||||
|         foreach ($parts as $part) { | ||||
|             $part      = trim($part); | ||||
|             $direction = 'asc'; | ||||
|             if ('-' === $part[0]) { | ||||
|                 $part      = substr($part, 1); | ||||
|                 $direction = 'desc'; | ||||
|             } | ||||
|             if (in_array($part, $this->allowedSort, true)) { | ||||
|                 $sortParameters[] = [$part, $direction]; | ||||
|             } | ||||
|         } | ||||
|         $bag->set('sort', $sortParameters); | ||||
| 
 | ||||
|         return $bag; | ||||
|         // return $this->getSortParameters($bag);
 | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -284,8 +232,6 @@ abstract class Controller extends BaseController | ||||
|         $baseUrl  = sprintf('%s/api/v1', request()->getSchemeAndHttpHost()); | ||||
|         $manager->setSerializer(new JsonApiSerializer($baseUrl)); | ||||
| 
 | ||||
|         // $transformer->collectMetaData(new Collection([$object]));
 | ||||
| 
 | ||||
|         $resource = new Item($object, $transformer, $key); | ||||
| 
 | ||||
|         return $manager->createData($resource)->toArray(); | ||||
|   | ||||
| @@ -26,6 +26,7 @@ namespace FireflyIII\Api\V1\Controllers\Data\Bulk; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Data\Bulk\TransactionRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Services\Internal\Destroy\AccountDestroyService; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -44,23 +45,23 @@ class TransactionController extends Controller | ||||
| { | ||||
|     private AccountRepositoryInterface $repository; | ||||
| 
 | ||||
|     protected array $acceptedRoles = [UserRoleEnum::MANAGE_TRANSACTIONS]; | ||||
| 
 | ||||
|     public function __construct() | ||||
|     { | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->repository = app(AccountRepositoryInterface::class); | ||||
|                 $this->repository->setUser(auth()->user()); | ||||
|                 $this->repository->setUserGroup($this->userGroup); | ||||
|                 $this->repository->setUser($this->user); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/bulkUpdateTransactions
 | ||||
|      */ | ||||
|     public function update(TransactionRequest $request): JsonResponse | ||||
|     { | ||||
|         $query  = $request->getAll(); | ||||
|   | ||||
| @@ -28,6 +28,7 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Data\DestroyRequest; | ||||
| use FireflyIII\Enums\AccountTypeEnum; | ||||
| use FireflyIII\Enums\TransactionTypeEnum; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Account; | ||||
| use FireflyIII\Models\TransactionJournal; | ||||
| @@ -55,16 +56,24 @@ class DestroyController extends Controller | ||||
| { | ||||
|     private bool $unused; | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/destroyData
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     protected array $acceptedRoles = [UserRoleEnum::FULL]; | ||||
| 
 | ||||
|     public function __construct() | ||||
|     { | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     public function destroy(DestroyRequest $request): JsonResponse | ||||
|     { | ||||
|         $objects         = $request->getObjects(); | ||||
|         $this->unused    = $request->boolean('unused', false); | ||||
|         $this->unused    = $request->boolean('unused'); | ||||
| 
 | ||||
|         $allExceptAssets = [AccountTypeEnum::BENEFICIARY->value, AccountTypeEnum::CASH->value, AccountTypeEnum::CREDITCARD->value, AccountTypeEnum::DEFAULT->value, AccountTypeEnum::EXPENSE->value, AccountTypeEnum::IMPORT->value, AccountTypeEnum::INITIAL_BALANCE->value, AccountTypeEnum::LIABILITY_CREDIT->value, AccountTypeEnum::RECONCILIATION->value, AccountTypeEnum::REVENUE->value]; | ||||
|         $all             = [AccountTypeEnum::ASSET->value, AccountTypeEnum::BENEFICIARY->value, AccountTypeEnum::CASH->value, AccountTypeEnum::CREDITCARD->value, AccountTypeEnum::DEBT->value, AccountTypeEnum::DEFAULT->value, AccountTypeEnum::EXPENSE->value, AccountTypeEnum::IMPORT->value, AccountTypeEnum::INITIAL_BALANCE->value, AccountTypeEnum::LIABILITY_CREDIT->value, AccountTypeEnum::LOAN->value, AccountTypeEnum::MORTGAGE->value, AccountTypeEnum::RECONCILIATION->value]; | ||||
|   | ||||
| @@ -26,9 +26,11 @@ namespace FireflyIII\Api\V1\Controllers\Data\Export; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Data\Export\ExportRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Support\Export\ExportDataGenerator; | ||||
| use Illuminate\Http\Response as LaravelResponse; | ||||
| use Safe\Exceptions\DatetimeException; | ||||
| 
 | ||||
| use function Safe\date; | ||||
| 
 | ||||
| @@ -38,6 +40,7 @@ use function Safe\date; | ||||
| class ExportController extends Controller | ||||
| { | ||||
|     private ExportDataGenerator $exporter; | ||||
|     protected array $acceptedRoles = [UserRoleEnum::READ_ONLY]; | ||||
| 
 | ||||
|     /** | ||||
|      * ExportController constructor. | ||||
| @@ -47,8 +50,10 @@ class ExportController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
|                 $this->exporter = app(ExportDataGenerator::class); | ||||
|                 $this->exporter->setUser(auth()->user()); | ||||
|                 $this->exporter->setUserGroup($this->userGroup); | ||||
|                 $this->exporter->setUser($this->user); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -56,9 +61,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportAccounts
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
| @@ -72,6 +74,7 @@ class ExportController extends Controller | ||||
| 
 | ||||
|     /** | ||||
|      * @throws FireflyException | ||||
|      * @throws DatetimeException | ||||
|      */ | ||||
|     private function returnExport(string $key): LaravelResponse | ||||
|     { | ||||
| @@ -97,9 +100,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportBills
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
| @@ -112,9 +112,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportBudgets
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
| @@ -127,9 +124,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportCategories
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
| @@ -142,9 +136,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportPiggies
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
| @@ -157,9 +148,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportRecurring
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
| @@ -172,9 +160,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportRules
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
| @@ -187,9 +172,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportTags
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * | ||||
|      * @SuppressWarnings("PHPMD.UnusedFormalParameter") | ||||
| @@ -202,9 +184,6 @@ class ExportController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/exportTransactions
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function transactions(ExportRequest $request): LaravelResponse | ||||
|   | ||||
| @@ -25,6 +25,7 @@ declare(strict_types=1); | ||||
| namespace FireflyIII\Api\V1\Controllers\Data; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\Account; | ||||
| use FireflyIII\Models\Bill; | ||||
| use FireflyIII\Models\Budget; | ||||
| @@ -44,10 +45,22 @@ use Illuminate\Http\JsonResponse; | ||||
|  */ | ||||
| class PurgeController extends Controller | ||||
| { | ||||
|     protected array $acceptedRoles = [UserRoleEnum::FULL]; | ||||
| 
 | ||||
|     public function __construct() | ||||
|     { | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->validateUserGroup($request); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * TODO cleanup and use repositories. | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/data/purgeData
 | ||||
|      */ | ||||
|     public function purge(): JsonResponse | ||||
|     { | ||||
| @@ -66,14 +79,6 @@ class PurgeController extends Controller | ||||
|         $repository = app(PiggyBankRepositoryInterface::class); | ||||
|         $repository->setUser($user); | ||||
|         $repository->purgeAll(); | ||||
|         //        $set  = PiggyBank::leftJoin('accounts', 'accounts.id', 'piggy_banks.account_id')
 | ||||
|         //            ->where('accounts.user_id', $user->id)->onlyTrashed()->get(['piggy_banks.*'])
 | ||||
|         //        ;
 | ||||
|         //
 | ||||
|         //        /** @var PiggyBank $piggy */
 | ||||
|         //        foreach ($set as $piggy) {
 | ||||
|         //            $piggy->forceDelete();
 | ||||
|         //        }
 | ||||
| 
 | ||||
|         // rule group
 | ||||
|         RuleGroup::whereUserId($user->id)->onlyTrashed()->forceDelete(); | ||||
|   | ||||
| @@ -64,10 +64,6 @@ class AccountController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseAsset
 | ||||
|      */ | ||||
|     public function asset(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
| @@ -91,10 +87,6 @@ class AccountController extends Controller | ||||
|         return response()->json($result); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseExpense
 | ||||
|      */ | ||||
|     public function expense(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start           = $request->getStart(); | ||||
|   | ||||
| @@ -58,20 +58,17 @@ class BillController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseBill
 | ||||
|      * | ||||
|      * Expenses per bill, possibly filtered by bill and account. | ||||
|      */ | ||||
|     public function bill(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $accounts        = $request->getAssetAccounts(); | ||||
|         $bills           = $request->getBills(); | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $response        = []; | ||||
|         $accounts         = $request->getAssetAccounts(); | ||||
|         $bills            = $request->getBills(); | ||||
|         $start            = $request->getStart(); | ||||
|         $end              = $request->getEnd(); | ||||
|         $convertToPrimary = Amount::convertToPrimary(); | ||||
|         $primary          = Amount::getPrimaryCurrency(); | ||||
|         $response         = []; | ||||
| 
 | ||||
|         // get all bills:
 | ||||
|         if (0 === $bills->count()) { | ||||
| @@ -79,25 +76,25 @@ class BillController extends Controller | ||||
|         } | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|         $collector        = app(GroupCollectorInterface::class); | ||||
|         $collector->setTypes([TransactionTypeEnum::WITHDRAWAL->value])->setRange($start, $end)->setSourceAccounts($accounts); | ||||
|         $collector->setBills($bills); | ||||
| 
 | ||||
|         $genericSet      = $collector->getExtractedJournals(); | ||||
|         $genericSet       = $collector->getExtractedJournals(); | ||||
|         foreach ($genericSet as $journal) { | ||||
|             $billId       = (int) $journal['bill_id']; | ||||
|             $currencyId   = (int) $journal['currency_id']; | ||||
|             $currencyCode = $journal['currency_code']; | ||||
|             $field        = 'amount'; | ||||
| 
 | ||||
|             // use the native amount if the user wants to convert to native currency
 | ||||
|             if ($convertToNative && $currencyId !== $default->id) { | ||||
|                 $currencyId   = $default->id; | ||||
|                 $currencyCode = $default->code; | ||||
|                 $field        = 'native_amount'; | ||||
|             // use the primary amount if the user wants to convert to primary currency
 | ||||
|             if ($convertToPrimary && $currencyId !== $primary->id) { | ||||
|                 $currencyId   = $primary->id; | ||||
|                 $currencyCode = $primary->code; | ||||
|                 $field        = 'pc_amount'; | ||||
|             } | ||||
|             // use foreign amount when the foreign currency IS the default currency.
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id && $default->id === $journal['foreign_currency_id']) { | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id && $primary->id === $journal['foreign_currency_id']) { | ||||
|                 $field = 'foreign_amount'; | ||||
|             } | ||||
|             Log::debug(sprintf('Journal #%d in bill #%d will use %s (%s %s)', $journal['transaction_group_id'], $billId, $field, $currencyCode, $journal[$field] ?? '0')); | ||||
| @@ -122,40 +119,37 @@ class BillController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseNoBill
 | ||||
|      * | ||||
|      * Expenses for no bill filtered by account. | ||||
|      */ | ||||
|     public function noBill(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $accounts        = $request->getAssetAccounts(); | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $response        = []; | ||||
|         $accounts         = $request->getAssetAccounts(); | ||||
|         $start            = $request->getStart(); | ||||
|         $end              = $request->getEnd(); | ||||
|         $convertToPrimary = Amount::convertToPrimary(); | ||||
|         $primary          = Amount::getPrimaryCurrency(); | ||||
|         $response         = []; | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|         $collector        = app(GroupCollectorInterface::class); | ||||
|         $collector->setTypes([TransactionTypeEnum::WITHDRAWAL->value])->setRange($start, $end)->setSourceAccounts($accounts); | ||||
|         $collector->withoutBill(); | ||||
| 
 | ||||
|         $genericSet      = $collector->getExtractedJournals(); | ||||
|         $genericSet       = $collector->getExtractedJournals(); | ||||
| 
 | ||||
|         foreach ($genericSet as $journal) { | ||||
|             $currencyId   = (int) $journal['currency_id']; | ||||
|             $currencyCode = $journal['currency_code']; | ||||
|             $field        = 'amount'; | ||||
| 
 | ||||
|             // use the native amount if the user wants to convert to native currency
 | ||||
|             if ($convertToNative && $currencyId !== $default->id) { | ||||
|                 $currencyId   = $default->id; | ||||
|                 $currencyCode = $default->code; | ||||
|                 $field        = 'native_amount'; | ||||
|             // use the primary amount if the user wants to convert to primary currency
 | ||||
|             if ($convertToPrimary && $currencyId !== $primary->id) { | ||||
|                 $currencyId   = $primary->id; | ||||
|                 $currencyCode = $primary->code; | ||||
|                 $field        = 'pc_amount'; | ||||
|             } | ||||
|             // use foreign amount when the foreign currency IS the default currency.
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id && $default->id === $journal['foreign_currency_id']) { | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id && $primary->id === $journal['foreign_currency_id']) { | ||||
|                 $field = 'foreign_amount'; | ||||
|             } | ||||
|             Log::debug(sprintf('Journal #%d will use %s (%s %s)', $journal['transaction_group_id'], $field, $currencyCode, $journal[$field] ?? '0')); | ||||
|   | ||||
| @@ -63,10 +63,6 @@ class BudgetController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseBudget
 | ||||
|      */ | ||||
|     public function budget(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
| @@ -80,7 +76,7 @@ class BudgetController extends Controller | ||||
| 
 | ||||
|         /** @var Budget $budget */ | ||||
|         foreach ($budgets as $budget) { | ||||
|             $expenses = $this->opsRepository->sumExpenses($start, $end, $assetAccounts, new Collection([$budget])); | ||||
|             $expenses = $this->opsRepository->sumExpenses($start, $end, $assetAccounts, new Collection()->push($budget)); | ||||
| 
 | ||||
|             /** @var array $expense */ | ||||
|             foreach ($expenses as $expense) { | ||||
| @@ -98,10 +94,6 @@ class BudgetController extends Controller | ||||
|         return response()->json($result); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseNoBudget
 | ||||
|      */ | ||||
|     public function noBudget(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
|   | ||||
| @@ -63,10 +63,6 @@ class CategoryController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightTransferCategory
 | ||||
|      */ | ||||
|     public function category(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
| @@ -80,7 +76,7 @@ class CategoryController extends Controller | ||||
| 
 | ||||
|         /** @var Category $category */ | ||||
|         foreach ($categories as $category) { | ||||
|             $expenses = $this->opsRepository->sumExpenses($start, $end, $assetAccounts, new Collection([$category])); | ||||
|             $expenses = $this->opsRepository->sumExpenses($start, $end, $assetAccounts, new Collection()->push($category)); | ||||
| 
 | ||||
|             /** @var array $expense */ | ||||
|             foreach ($expenses as $expense) { | ||||
| @@ -98,10 +94,6 @@ class CategoryController extends Controller | ||||
|         return response()->json($result); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightTransferNoCategory
 | ||||
|      */ | ||||
|     public function noCategory(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
|   | ||||
| @@ -37,41 +37,37 @@ use Illuminate\Support\Facades\Log; | ||||
|  */ | ||||
| class PeriodController extends Controller | ||||
| { | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseTotal
 | ||||
|      */ | ||||
|     public function total(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $accounts        = $request->getAssetAccounts(); | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $accounts         = $request->getAssetAccounts(); | ||||
|         $start            = $request->getStart(); | ||||
|         $end              = $request->getEnd(); | ||||
|         $response         = []; | ||||
|         $convertToPrimary = Amount::convertToPrimary(); | ||||
|         $primary          = Amount::getPrimaryCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type)
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|         $collector        = app(GroupCollectorInterface::class); | ||||
|         $collector->setTypes([TransactionTypeEnum::WITHDRAWAL->value])->setRange($start, $end)->setSourceAccounts($accounts); | ||||
|         $genericSet      = $collector->getExtractedJournals(); | ||||
|         $genericSet       = $collector->getExtractedJournals(); | ||||
|         foreach ($genericSet as $journal) { | ||||
|             // same code as many other sumExpense methods. I think this needs some kind of generic method.
 | ||||
|             $amount                                    = '0'; | ||||
|             $currencyId                                = (int) $journal['currency_id']; | ||||
|             $currencyCode                              = $journal['currency_code']; | ||||
|             if ($convertToNative) { | ||||
|             if ($convertToPrimary) { | ||||
|                 $amount = Amount::getAmountFromJournal($journal); | ||||
|                 if ($default->id !== (int) $journal['currency_id'] && $default->id !== (int) $journal['foreign_currency_id']) { | ||||
|                     $currencyId   = $default->id; | ||||
|                     $currencyCode = $default->code; | ||||
|                 if ($primary->id !== (int) $journal['currency_id'] && $primary->id !== (int) $journal['foreign_currency_id']) { | ||||
|                     $currencyId   = $primary->id; | ||||
|                     $currencyCode = $primary->code; | ||||
|                 } | ||||
|                 if ($default->id !== (int) $journal['currency_id'] && $default->id === (int) $journal['foreign_currency_id']) { | ||||
|                 if ($primary->id !== (int) $journal['currency_id'] && $primary->id === (int) $journal['foreign_currency_id']) { | ||||
|                     $currencyId   = $journal['foreign_currency_id']; | ||||
|                     $currencyCode = $journal['foreign_currency_code']; | ||||
|                 } | ||||
|                 Log::debug(sprintf('[a] Add amount %s %s', $currencyCode, $amount)); | ||||
|             } | ||||
|             if (!$convertToNative) { | ||||
|             if (!$convertToPrimary) { | ||||
|                 // ignore the amount in foreign currency.
 | ||||
|                 Log::debug(sprintf('[b] Add amount %s %s', $currencyCode, $journal['amount'])); | ||||
|                 $amount = $journal['amount']; | ||||
|   | ||||
| @@ -57,45 +57,42 @@ class TagController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseNoTag
 | ||||
|      * | ||||
|      * Expenses for no tag filtered by account. | ||||
|      */ | ||||
|     public function noTag(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $accounts        = $request->getAssetAccounts(); | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $accounts         = $request->getAssetAccounts(); | ||||
|         $start            = $request->getStart(); | ||||
|         $end              = $request->getEnd(); | ||||
|         $response         = []; | ||||
|         $convertToPrimary = Amount::convertToPrimary(); | ||||
|         $primary          = Amount::getPrimaryCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|         $collector        = app(GroupCollectorInterface::class); | ||||
|         $collector->setTypes([TransactionTypeEnum::WITHDRAWAL->value])->setRange($start, $end)->setSourceAccounts($accounts); | ||||
|         $collector->withoutTags(); | ||||
| 
 | ||||
|         $genericSet      = $collector->getExtractedJournals(); | ||||
|         $genericSet       = $collector->getExtractedJournals(); | ||||
| 
 | ||||
|         foreach ($genericSet as $journal) { | ||||
|             // same code as many other sumExpense methods. I think this needs some kind of generic method.
 | ||||
|             $amount                                    = '0'; | ||||
|             $currencyId                                = (int) $journal['currency_id']; | ||||
|             $currencyCode                              = $journal['currency_code']; | ||||
|             if ($convertToNative) { | ||||
|             if ($convertToPrimary) { | ||||
|                 $amount = Amount::getAmountFromJournal($journal); | ||||
|                 if ($default->id !== (int) $journal['currency_id'] && $default->id !== (int) $journal['foreign_currency_id']) { | ||||
|                     $currencyId   = $default->id; | ||||
|                     $currencyCode = $default->code; | ||||
|                 if ($primary->id !== (int) $journal['currency_id'] && $primary->id !== (int) $journal['foreign_currency_id']) { | ||||
|                     $currencyId   = $primary->id; | ||||
|                     $currencyCode = $primary->code; | ||||
|                 } | ||||
|                 if ($default->id !== (int) $journal['currency_id'] && $default->id === (int) $journal['foreign_currency_id']) { | ||||
|                 if ($primary->id !== (int) $journal['currency_id'] && $primary->id === (int) $journal['foreign_currency_id']) { | ||||
|                     $currencyId   = $journal['foreign_currency_id']; | ||||
|                     $currencyCode = $journal['foreign_currency_code']; | ||||
|                 } | ||||
|                 Log::debug(sprintf('[a] Add amount %s %s', $currencyCode, $amount)); | ||||
|             } | ||||
|             if (!$convertToNative) { | ||||
|             if (!$convertToPrimary) { | ||||
|                 // ignore the amount in foreign currency.
 | ||||
|                 Log::debug(sprintf('[b] Add amount %s %s', $currencyCode, $journal['amount'])); | ||||
|                 $amount = $journal['amount']; | ||||
| @@ -115,9 +112,6 @@ class TagController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightExpenseTag
 | ||||
|      * | ||||
|      * Expenses per tag, possibly filtered by tag and account. | ||||
|      */ | ||||
|     public function tag(GenericRequest $request): JsonResponse | ||||
|   | ||||
| @@ -64,10 +64,6 @@ class AccountController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightIncomeAsset
 | ||||
|      */ | ||||
|     public function asset(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
| @@ -92,10 +88,6 @@ class AccountController extends Controller | ||||
|         return response()->json($result); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightIncomeRevenue
 | ||||
|      */ | ||||
|     public function revenue(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start           = $request->getStart(); | ||||
|   | ||||
| @@ -63,10 +63,6 @@ class CategoryController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightIncomeCategory
 | ||||
|      */ | ||||
|     public function category(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
| @@ -80,7 +76,7 @@ class CategoryController extends Controller | ||||
| 
 | ||||
|         /** @var Category $category */ | ||||
|         foreach ($categories as $category) { | ||||
|             $expenses = $this->opsRepository->sumIncome($start, $end, $assetAccounts, new Collection([$category])); | ||||
|             $expenses = $this->opsRepository->sumIncome($start, $end, $assetAccounts, new Collection()->push($category)); | ||||
| 
 | ||||
|             /** @var array $expense */ | ||||
|             foreach ($expenses as $expense) { | ||||
| @@ -98,10 +94,6 @@ class CategoryController extends Controller | ||||
|         return response()->json($result); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightIncomeNoCategory
 | ||||
|      */ | ||||
|     public function noCategory(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
|   | ||||
| @@ -29,6 +29,7 @@ use FireflyIII\Api\V1\Requests\Insight\GenericRequest; | ||||
| use FireflyIII\Enums\TransactionTypeEnum; | ||||
| use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -36,36 +37,32 @@ use Illuminate\Http\JsonResponse; | ||||
|  */ | ||||
| class PeriodController extends Controller | ||||
| { | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightIncomeTotal
 | ||||
|      */ | ||||
|     public function total(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $accounts        = $request->getAssetAccounts(); | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $accounts         = $request->getAssetAccounts(); | ||||
|         $start            = $request->getStart(); | ||||
|         $end              = $request->getEnd(); | ||||
|         $response         = []; | ||||
|         $convertToPrimary = Amount::convertToPrimary(); | ||||
|         $primary          = Amount::getPrimaryCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type)
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|         $collector        = app(GroupCollectorInterface::class); | ||||
|         $collector->setTypes([TransactionTypeEnum::DEPOSIT->value])->setRange($start, $end)->setDestinationAccounts($accounts); | ||||
|         $genericSet      = $collector->getExtractedJournals(); | ||||
|         $genericSet       = $collector->getExtractedJournals(); | ||||
|         foreach ($genericSet as $journal) { | ||||
|             // currency
 | ||||
|             $currencyId                                = $journal['currency_id']; | ||||
|             $currencyCode                              = $journal['currency_code']; | ||||
|             $field                                     = $convertToNative && $currencyId !== $default->id ? 'native_amount' : 'amount'; | ||||
|             $field                                     = $convertToPrimary && $currencyId !== $primary->id ? 'pc_amount' : 'amount'; | ||||
| 
 | ||||
|             // perhaps use default currency instead?
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id) { | ||||
|                 $currencyId   = $default->id; | ||||
|                 $currencyCode = $default->code; | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id) { | ||||
|                 $currencyId   = $primary->id; | ||||
|                 $currencyCode = $primary->code; | ||||
|             } | ||||
|             // use foreign amount when the foreign currency IS the default currency.
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id && $default->id === $journal['foreign_currency_id']) { | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id && $primary->id === $journal['foreign_currency_id']) { | ||||
|                 $field = 'foreign_amount'; | ||||
|             } | ||||
| 
 | ||||
| @@ -75,7 +72,7 @@ class PeriodController extends Controller | ||||
|                 'currency_id'      => (string) $currencyId, | ||||
|                 'currency_code'    => $currencyCode, | ||||
|             ]; | ||||
|             $response[$currencyId]['difference']       = bcadd($response[$currencyId]['difference'], (string) app('steam')->positive($journal[$field])); | ||||
|             $response[$currencyId]['difference']       = bcadd($response[$currencyId]['difference'], Steam::positive($journal[$field])); | ||||
|             $response[$currencyId]['difference_float'] = (float) $response[$currencyId]['difference']; // float but on purpose.
 | ||||
|         } | ||||
| 
 | ||||
|   | ||||
| @@ -30,6 +30,7 @@ use FireflyIII\Enums\TransactionTypeEnum; | ||||
| use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Repositories\Tag\TagRepositoryInterface; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -57,40 +58,37 @@ class TagController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightIncomeTag
 | ||||
|      * | ||||
|      * Expenses for no tag filtered by account. | ||||
|      */ | ||||
|     public function noTag(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $accounts        = $request->getAssetAccounts(); | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $accounts         = $request->getAssetAccounts(); | ||||
|         $start            = $request->getStart(); | ||||
|         $end              = $request->getEnd(); | ||||
|         $response         = []; | ||||
|         $convertToPrimary = Amount::convertToPrimary(); | ||||
|         $primary          = Amount::getPrimaryCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|         $collector        = app(GroupCollectorInterface::class); | ||||
|         $collector->setTypes([TransactionTypeEnum::DEPOSIT->value])->setRange($start, $end)->setDestinationAccounts($accounts); | ||||
|         $collector->withoutTags(); | ||||
| 
 | ||||
|         $genericSet      = $collector->getExtractedJournals(); | ||||
|         $genericSet       = $collector->getExtractedJournals(); | ||||
| 
 | ||||
|         foreach ($genericSet as $journal) { | ||||
|             // currency
 | ||||
|             $currencyId                                = $journal['currency_id']; | ||||
|             $currencyCode                              = $journal['currency_code']; | ||||
|             $field                                     = $convertToNative && $currencyId !== $default->id ? 'native_amount' : 'amount'; | ||||
|             $field                                     = $convertToPrimary && $currencyId !== $primary->id ? 'pc_amount' : 'amount'; | ||||
| 
 | ||||
|             // perhaps use default currency instead?
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id) { | ||||
|                 $currencyId   = $default->id; | ||||
|                 $currencyCode = $default->code; | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id) { | ||||
|                 $currencyId   = $primary->id; | ||||
|                 $currencyCode = $primary->code; | ||||
|             } | ||||
|             // use foreign amount when the foreign currency IS the default currency.
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id && $default->id === $journal['foreign_currency_id']) { | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id && $primary->id === $journal['foreign_currency_id']) { | ||||
|                 $field = 'foreign_amount'; | ||||
|             } | ||||
| 
 | ||||
| @@ -100,7 +98,7 @@ class TagController extends Controller | ||||
|                 'currency_id'      => (string) $currencyId, | ||||
|                 'currency_code'    => $currencyCode, | ||||
|             ]; | ||||
|             $response[$currencyId]['difference']       = bcadd($response[$currencyId]['difference'], (string) app('steam')->positive($journal[$field])); | ||||
|             $response[$currencyId]['difference']       = bcadd($response[$currencyId]['difference'], Steam::positive($journal[$field])); | ||||
|             $response[$currencyId]['difference_float'] = (float) $response[$currencyId]['difference']; | ||||
| 
 | ||||
|         } | ||||
| @@ -109,9 +107,6 @@ class TagController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightIncomeNoTag
 | ||||
|      * | ||||
|      * Expenses per tag, possibly filtered by tag and account. | ||||
|      */ | ||||
|     public function tag(GenericRequest $request): JsonResponse | ||||
| @@ -154,7 +149,7 @@ class TagController extends Controller | ||||
|                         'currency_id'      => (string) $currencyId, | ||||
|                         'currency_code'    => $journal['currency_code'], | ||||
|                     ]; | ||||
|                     $response[$key]['difference']       = bcadd((string) $response[$key]['difference'], (string) app('steam')->positive($journal['amount'])); | ||||
|                     $response[$key]['difference']       = bcadd((string) $response[$key]['difference'], Steam::positive($journal['amount'])); | ||||
|                     $response[$key]['difference_float'] = (float) $response[$key]['difference']; | ||||
|                 } | ||||
| 
 | ||||
| @@ -166,10 +161,7 @@ class TagController extends Controller | ||||
|                         'currency_id'      => (string) $foreignCurrencyId, | ||||
|                         'currency_code'    => $journal['foreign_currency_code'], | ||||
|                     ]; | ||||
|                     $response[$foreignKey]['difference']       = bcadd( | ||||
|                         (string) $response[$foreignKey]['difference'], | ||||
|                         (string) app('steam')->positive($journal['foreign_amount']) | ||||
|                     ); | ||||
|                     $response[$foreignKey]['difference']       = bcadd((string) $response[$foreignKey]['difference'], Steam::positive($journal['foreign_amount'])); | ||||
|                     $response[$foreignKey]['difference_float'] = (float) $response[$foreignKey]['difference']; | ||||
|                 } | ||||
|             } | ||||
|   | ||||
| @@ -56,10 +56,6 @@ class AccountController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightTransfers
 | ||||
|      */ | ||||
|     public function asset(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
|   | ||||
| @@ -63,10 +63,6 @@ class CategoryController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightTransferCategory
 | ||||
|      */ | ||||
|     public function category(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
| @@ -80,7 +76,7 @@ class CategoryController extends Controller | ||||
| 
 | ||||
|         /** @var Category $category */ | ||||
|         foreach ($categories as $category) { | ||||
|             $expenses = $this->opsRepository->sumTransfers($start, $end, $assetAccounts, new Collection([$category])); | ||||
|             $expenses = $this->opsRepository->sumTransfers($start, $end, $assetAccounts, new Collection()->push($category)); | ||||
| 
 | ||||
|             /** @var array $expense */ | ||||
|             foreach ($expenses as $expense) { | ||||
| @@ -98,10 +94,6 @@ class CategoryController extends Controller | ||||
|         return response()->json($result); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightTransferNoCategory
 | ||||
|      */ | ||||
|     public function noCategory(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $start         = $request->getStart(); | ||||
|   | ||||
| @@ -29,6 +29,7 @@ use FireflyIII\Api\V1\Requests\Insight\GenericRequest; | ||||
| use FireflyIII\Enums\TransactionTypeEnum; | ||||
| use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -36,36 +37,32 @@ use Illuminate\Http\JsonResponse; | ||||
|  */ | ||||
| class PeriodController extends Controller | ||||
| { | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightTransferTotal
 | ||||
|      */ | ||||
|     public function total(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $accounts        = $request->getAssetAccounts(); | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $accounts         = $request->getAssetAccounts(); | ||||
|         $start            = $request->getStart(); | ||||
|         $end              = $request->getEnd(); | ||||
|         $response         = []; | ||||
|         $convertToPrimary = Amount::convertToPrimary(); | ||||
|         $primary          = Amount::getPrimaryCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type)
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|         $collector        = app(GroupCollectorInterface::class); | ||||
|         $collector->setTypes([TransactionTypeEnum::TRANSFER->value])->setRange($start, $end)->setDestinationAccounts($accounts); | ||||
|         $genericSet      = $collector->getExtractedJournals(); | ||||
|         $genericSet       = $collector->getExtractedJournals(); | ||||
|         foreach ($genericSet as $journal) { | ||||
|             // currency
 | ||||
|             $currencyId                                = $journal['currency_id']; | ||||
|             $currencyCode                              = $journal['currency_code']; | ||||
|             $field                                     = $convertToNative && $currencyId !== $default->id ? 'native_amount' : 'amount'; | ||||
|             $field                                     = $convertToPrimary && $currencyId !== $primary->id ? 'pc_amount' : 'amount'; | ||||
| 
 | ||||
|             // perhaps use default currency instead?
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id) { | ||||
|                 $currencyId   = $default->id; | ||||
|                 $currencyCode = $default->code; | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id) { | ||||
|                 $currencyId   = $primary->id; | ||||
|                 $currencyCode = $primary->code; | ||||
|             } | ||||
|             // use foreign amount when the foreign currency IS the default currency.
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id && $default->id === $journal['foreign_currency_id']) { | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id && $primary->id === $journal['foreign_currency_id']) { | ||||
|                 $field = 'foreign_amount'; | ||||
|             } | ||||
| 
 | ||||
| @@ -75,7 +72,7 @@ class PeriodController extends Controller | ||||
|                 'currency_id'      => (string) $currencyId, | ||||
|                 'currency_code'    => $currencyCode, | ||||
|             ]; | ||||
|             $response[$currencyId]['difference']       = bcadd($response[$currencyId]['difference'], (string) app('steam')->positive($journal[$field])); | ||||
|             $response[$currencyId]['difference']       = bcadd($response[$currencyId]['difference'], Steam::positive($journal[$field])); | ||||
|             $response[$currencyId]['difference_float'] = (float) $response[$currencyId]['difference']; | ||||
| 
 | ||||
|         } | ||||
|   | ||||
| @@ -30,6 +30,7 @@ use FireflyIII\Enums\TransactionTypeEnum; | ||||
| use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Repositories\Tag\TagRepositoryInterface; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
| @@ -56,40 +57,36 @@ class TagController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightTransferNoTag
 | ||||
|      */ | ||||
|     public function noTag(GenericRequest $request): JsonResponse | ||||
|     { | ||||
|         $accounts        = $request->getAssetAccounts(); | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $accounts         = $request->getAssetAccounts(); | ||||
|         $start            = $request->getStart(); | ||||
|         $end              = $request->getEnd(); | ||||
|         $response         = []; | ||||
|         $convertToPrimary = Amount::convertToPrimary(); | ||||
|         $primary          = Amount::getPrimaryCurrency(); | ||||
| 
 | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|         $collector        = app(GroupCollectorInterface::class); | ||||
|         $collector->setTypes([TransactionTypeEnum::TRANSFER->value])->setRange($start, $end)->setDestinationAccounts($accounts); | ||||
|         $collector->withoutTags(); | ||||
| 
 | ||||
|         $genericSet      = $collector->getExtractedJournals(); | ||||
|         $genericSet       = $collector->getExtractedJournals(); | ||||
| 
 | ||||
|         foreach ($genericSet as $journal) { | ||||
|             // currency
 | ||||
|             $currencyId                                = $journal['currency_id']; | ||||
|             $currencyCode                              = $journal['currency_code']; | ||||
|             $field                                     = $convertToNative && $currencyId !== $default->id ? 'native_amount' : 'amount'; | ||||
|             $field                                     = $convertToPrimary && $currencyId !== $primary->id ? 'pc_amount' : 'amount'; | ||||
| 
 | ||||
|             // perhaps use default currency instead?
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id) { | ||||
|                 $currencyId   = $default->id; | ||||
|                 $currencyCode = $default->code; | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id) { | ||||
|                 $currencyId   = $primary->id; | ||||
|                 $currencyCode = $primary->code; | ||||
|             } | ||||
|             // use foreign amount when the foreign currency IS the default currency.
 | ||||
|             if ($convertToNative && $journal['currency_id'] !== $default->id && $default->id === $journal['foreign_currency_id']) { | ||||
|             if ($convertToPrimary && $journal['currency_id'] !== $primary->id && $primary->id === $journal['foreign_currency_id']) { | ||||
|                 $field = 'foreign_amount'; | ||||
|             } | ||||
| 
 | ||||
| @@ -99,7 +96,7 @@ class TagController extends Controller | ||||
|                 'currency_id'      => (string) $currencyId, | ||||
|                 'currency_code'    => $currencyCode, | ||||
|             ]; | ||||
|             $response[$currencyId]['difference']       = bcadd($response[$currencyId]['difference'], (string) app('steam')->positive($journal[$field])); | ||||
|             $response[$currencyId]['difference']       = bcadd($response[$currencyId]['difference'], Steam::positive($journal[$field])); | ||||
|             $response[$currencyId]['difference_float'] = (float) $response[$currencyId]['difference']; | ||||
| 
 | ||||
|         } | ||||
| @@ -108,9 +105,6 @@ class TagController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/insight/insightTransferTag
 | ||||
|      * | ||||
|      * Transfers per tag, possibly filtered by tag and account. | ||||
|      */ | ||||
|     public function tag(GenericRequest $request): JsonResponse | ||||
| @@ -153,7 +147,7 @@ class TagController extends Controller | ||||
|                         'currency_id'      => (string) $currencyId, | ||||
|                         'currency_code'    => $journal['currency_code'], | ||||
|                     ]; | ||||
|                     $response[$key]['difference']       = bcadd((string) $response[$key]['difference'], (string) app('steam')->positive($journal['amount'])); | ||||
|                     $response[$key]['difference']       = bcadd((string) $response[$key]['difference'], Steam::positive($journal['amount'])); | ||||
|                     $response[$key]['difference_float'] = (float) $response[$key]['difference']; | ||||
|                 } | ||||
| 
 | ||||
| @@ -167,7 +161,7 @@ class TagController extends Controller | ||||
|                     ]; | ||||
|                     $response[$foreignKey]['difference']       = bcadd( | ||||
|                         (string) $response[$foreignKey]['difference'], | ||||
|                         (string) app('steam')->positive($journal['foreign_amount']) | ||||
|                         Steam::positive($journal['foreign_amount']) | ||||
|                     ); | ||||
|                     $response[$foreignKey]['difference_float'] = (float) $response[$foreignKey]['difference']; // intentional float
 | ||||
|                 } | ||||
|   | ||||
| @@ -55,9 +55,6 @@ class DestroyController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/accounts/deleteAccount
 | ||||
|      * | ||||
|      * Remove the specified resource from storage. | ||||
|      */ | ||||
|     public function destroy(Account $account): JsonResponse | ||||
|   | ||||
| @@ -30,6 +30,7 @@ use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Models\Account; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\TransactionFilter; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\PiggyBankEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\TransactionGroupEnrichment; | ||||
| use FireflyIII\Transformers\AttachmentTransformer; | ||||
| use FireflyIII\Transformers\PiggyBankTransformer; | ||||
| @@ -70,9 +71,6 @@ class ListController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/accounts/listAttachmentByAccount
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function attachments(Account $account): JsonResponse | ||||
| @@ -99,9 +97,6 @@ class ListController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/accounts/listPiggyBankByAccount
 | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function piggyBanks(Account $account): JsonResponse | ||||
| @@ -117,6 +112,13 @@ class ListController extends Controller | ||||
|         $count       = $collection->count(); | ||||
|         $piggyBanks  = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new PiggyBankEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $piggyBanks  = $enrichment->enrich($piggyBanks); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($piggyBanks, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.accounts.piggy-banks', [$account->id]).$this->buildParams()); | ||||
| @@ -125,16 +127,13 @@ class ListController extends Controller | ||||
|         $transformer = app(PiggyBankTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new FractalCollection($piggyBanks, $transformer, 'piggy_banks'); | ||||
|         $resource    = new FractalCollection($piggyBanks, $transformer, 'piggy-banks'); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/accounts/listTransactionByAccount
 | ||||
|      * | ||||
|      * Show all transaction groups related to the account. | ||||
|      * | ||||
|      * @throws FireflyException | ||||
| @@ -153,7 +152,7 @@ class ListController extends Controller | ||||
|         // use new group collector:
 | ||||
|         /** @var GroupCollectorInterface $collector */ | ||||
|         $collector    = app(GroupCollectorInterface::class); | ||||
|         $collector->setUser($admin)->setAccounts(new Collection([$account])) | ||||
|         $collector->setUser($admin)->setAccounts(new Collection()->push($account)) | ||||
|             ->withAPIInformation()->setLimit($pageSize)->setPage($this->parameters->get('page'))->setTypes($types) | ||||
|         ; | ||||
| 
 | ||||
|   | ||||
| @@ -25,6 +25,7 @@ declare(strict_types=1); | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\Account; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\Account\ShowRequest; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Account; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| @@ -33,7 +34,6 @@ use FireflyIII\Support\JsonApi\Enrichments\AccountEnrichment; | ||||
| use FireflyIII\Transformers\AccountTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Http\Request; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| use League\Fractal\Resource\Collection as FractalCollection; | ||||
| @@ -67,41 +67,42 @@ class ShowController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/accounts/listAccount
 | ||||
|      * | ||||
|      * Display a listing of the resource. | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function index(Request $request): JsonResponse | ||||
|     public function index(ShowRequest $request): JsonResponse | ||||
|     { | ||||
|         $manager     = $this->getManager(); | ||||
|         $type        = $request->get('type') ?? 'all'; | ||||
|         $this->parameters->set('type', $type); | ||||
|         $params      = $request->getParameters(); | ||||
|         $this->parameters->set('type', $params['type']); | ||||
| 
 | ||||
|         // types to get, page size:
 | ||||
|         $types       = $this->mapAccountTypes($this->parameters->get('type')); | ||||
|         $pageSize    = $this->parameters->get('limit'); | ||||
|         $types       = $this->mapAccountTypes($params['type']); | ||||
| 
 | ||||
|         // get list of accounts. Count it and split it.
 | ||||
|         $this->repository->resetAccountOrder(); | ||||
|         $collection  = $this->repository->getAccountsByType($types, $this->parameters->get('sort') ?? []); | ||||
|         $collection  = $this->repository->getAccountsByType($types, $params['sort']); | ||||
|         $count       = $collection->count(); | ||||
| 
 | ||||
|         // continue sort:
 | ||||
|         $accounts    = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
|         // TODO if the user sorts on DB dependent field there must be no slice before enrichment, only after.
 | ||||
|         // TODO still need to figure out how to do this easily.
 | ||||
|         $accounts    = $collection->slice(($this->parameters->get('page') - 1) * $params['limit'], $params['limit']); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new AccountEnrichment(); | ||||
|         $enrichment->setSort($params['sort']); | ||||
|         $enrichment->setDate($this->parameters->get('date')); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setNative($this->nativeCurrency); | ||||
|         $accounts    = $enrichment->enrich($accounts); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($accounts, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator   = new LengthAwarePaginator($accounts, $count, $params['limit'], $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.accounts.index').$this->buildParams()); | ||||
| 
 | ||||
|         /** @var AccountTransformer $transformer */ | ||||
| @@ -120,7 +121,7 @@ class ShowController extends Controller | ||||
|      * | ||||
|      * Show single instance. | ||||
|      */ | ||||
|     public function show(Account $account): JsonResponse | ||||
|     public function show(ShowRequest $request, Account $account): JsonResponse | ||||
|     { | ||||
|         // get list of accounts. Count it and split it.
 | ||||
|         $this->repository->resetAccountOrder(); | ||||
| @@ -131,8 +132,10 @@ class ShowController extends Controller | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new AccountEnrichment(); | ||||
|         $enrichment->setDate($this->parameters->get('date')); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setNative($this->nativeCurrency); | ||||
|         $account     = $enrichment->enrichSingle($account); | ||||
| 
 | ||||
| 
 | ||||
|   | ||||
| @@ -75,8 +75,8 @@ class StoreController extends Controller | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new AccountEnrichment(); | ||||
|         $enrichment->setDate(null); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setNative($this->nativeCurrency); | ||||
|         $account     = $enrichment->enrichSingle($account); | ||||
| 
 | ||||
|         /** @var AccountTransformer $transformer */ | ||||
|   | ||||
| @@ -80,8 +80,8 @@ class UpdateController extends Controller | ||||
|         /** @var User $admin */ | ||||
|         $admin        = auth()->user(); | ||||
|         $enrichment   = new AccountEnrichment(); | ||||
|         $enrichment->setDate(null); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setNative($this->nativeCurrency); | ||||
|         $account      = $enrichment->enrichSingle($account); | ||||
| 
 | ||||
|         /** @var AccountTransformer $transformer */ | ||||
|   | ||||
| @@ -28,6 +28,7 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\AvailableBudget; | ||||
| use FireflyIII\Repositories\Budget\AvailableBudgetRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\AvailableBudgetEnrichment; | ||||
| use FireflyIII\Transformers\AvailableBudgetTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -75,7 +76,6 @@ class ShowController extends Controller | ||||
| 
 | ||||
|         // types to get, page size:
 | ||||
|         $pageSize         = $this->parameters->get('limit'); | ||||
| 
 | ||||
|         $start            = $this->parameters->get('start'); | ||||
|         $end              = $this->parameters->get('end'); | ||||
| 
 | ||||
| @@ -84,6 +84,13 @@ class ShowController extends Controller | ||||
|         $count            = $collection->count(); | ||||
|         $availableBudgets = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin            = auth()->user(); | ||||
|         $enrichment       = new AvailableBudgetEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $availableBudgets = $enrichment->enrich($availableBudgets); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator        = new LengthAwarePaginator($availableBudgets, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.available-budgets.index').$this->buildParams()); | ||||
| @@ -106,13 +113,25 @@ class ShowController extends Controller | ||||
|      */ | ||||
|     public function show(AvailableBudget $availableBudget): JsonResponse | ||||
|     { | ||||
|         $manager     = $this->getManager(); | ||||
|         $manager         = $this->getManager(); | ||||
|         //        $start           = $this->parameters->get('start');
 | ||||
|         //        $end             = $this->parameters->get('end');
 | ||||
| 
 | ||||
|         /** @var AvailableBudgetTransformer $transformer */ | ||||
|         $transformer = app(AvailableBudgetTransformer::class); | ||||
|         $transformer     = app(AvailableBudgetTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new Item($availableBudget, $transformer, 'available_budgets'); | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin           = auth()->user(); | ||||
|         $enrichment      = new AvailableBudgetEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         //        $enrichment->setStart($start);
 | ||||
|         //        $enrichment->setEnd($end);
 | ||||
|         $availableBudget = $enrichment->enrichSingle($availableBudget); | ||||
| 
 | ||||
| 
 | ||||
|         $resource        = new Item($availableBudget, $transformer, 'available_budgets'); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|     } | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Bill; | ||||
| use FireflyIII\Repositories\Bill\BillRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\SubscriptionEnrichment; | ||||
| use FireflyIII\Transformers\BillTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| @@ -76,6 +78,15 @@ class ShowController extends Controller | ||||
|         $bills       = $bills->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
|         $paginator   = new LengthAwarePaginator($bills, $count, $pageSize, $this->parameters->get('page')); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new SubscriptionEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $bills       = $enrichment->enrich($bills); | ||||
| 
 | ||||
|         /** @var BillTransformer $transformer */ | ||||
|         $transformer = app(BillTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| @@ -96,6 +107,15 @@ class ShowController extends Controller | ||||
|     { | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new SubscriptionEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $bill        = $enrichment->enrichSingle($bill); | ||||
| 
 | ||||
|         /** @var BillTransformer $transformer */ | ||||
|         $transformer = app(BillTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -29,7 +29,9 @@ use FireflyIII\Api\V1\Requests\Models\Bill\StoreRequest; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Repositories\Bill\BillRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\TransactionFilter; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\SubscriptionEnrichment; | ||||
| use FireflyIII\Transformers\BillTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -72,6 +74,15 @@ class StoreController extends Controller | ||||
|         $bill        = $this->repository->store($data); | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new SubscriptionEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $bill        = $enrichment->enrichSingle($bill); | ||||
| 
 | ||||
|         /** @var BillTransformer $transformer */ | ||||
|         $transformer = app(BillTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\Bill\UpdateRequest; | ||||
| use FireflyIII\Models\Bill; | ||||
| use FireflyIII\Repositories\Bill\BillRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\SubscriptionEnrichment; | ||||
| use FireflyIII\Transformers\BillTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -67,6 +69,15 @@ class UpdateController extends Controller | ||||
|         $bill        = $this->repository->update($bill, $data); | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new SubscriptionEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $bill        = $enrichment->enrichSingle($bill); | ||||
| 
 | ||||
|         /** @var BillTransformer $transformer */ | ||||
|         $transformer = app(BillTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -32,6 +32,7 @@ use FireflyIII\Models\Budget; | ||||
| use FireflyIII\Repositories\Budget\BudgetLimitRepositoryInterface; | ||||
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\TransactionFilter; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetLimitEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\TransactionGroupEnrichment; | ||||
| use FireflyIII\Transformers\AttachmentTransformer; | ||||
| use FireflyIII\Transformers\BudgetLimitTransformer; | ||||
| @@ -117,6 +118,14 @@ class ListController extends Controller | ||||
|         $paginator    = new LengthAwarePaginator($budgetLimits, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.budgets.budget-limits', [$budget->id]).$this->buildParams()); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin        = auth()->user(); | ||||
|         $enrichment   = new BudgetLimitEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budgetLimits = $enrichment->enrich($budgetLimits); | ||||
| 
 | ||||
| 
 | ||||
|         /** @var BudgetLimitTransformer $transformer */ | ||||
|         $transformer  = app(BudgetLimitTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -29,7 +29,9 @@ use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Budget; | ||||
| use FireflyIII\Repositories\Budget\BudgetLimitRepositoryInterface; | ||||
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetEnrichment; | ||||
| use FireflyIII\Transformers\BudgetTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| @@ -82,6 +84,15 @@ class ShowController extends Controller | ||||
|         $count       = $collection->count(); | ||||
|         $budgets     = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new BudgetEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $budgets     = $enrichment->enrich($budgets); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($budgets, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.budgets.index').$this->buildParams()); | ||||
| @@ -103,6 +114,15 @@ class ShowController extends Controller | ||||
|     { | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new BudgetEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $budget      = $enrichment->enrichSingle($budget); | ||||
| 
 | ||||
|         /** @var BudgetTransformer $transformer */ | ||||
|         $transformer = app(BudgetTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\Budget\StoreRequest; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetEnrichment; | ||||
| use FireflyIII\Transformers\BudgetTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -65,10 +67,19 @@ class StoreController extends Controller | ||||
|      */ | ||||
|     public function store(StoreRequest $request): JsonResponse | ||||
|     { | ||||
|         $budget      = $this->repository->store($request->getAll()); | ||||
|         $data        = $request->getAll(); | ||||
|         $data['fire_webhooks'] ??= true; | ||||
|         $budget      = $this->repository->store($data); | ||||
|         $budget->refresh(); | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new BudgetEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budget      = $enrichment->enrichSingle($budget); | ||||
| 
 | ||||
|         /** @var BudgetTransformer $transformer */ | ||||
|         $transformer = app(BudgetTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\Budget\UpdateRequest; | ||||
| use FireflyIII\Models\Budget; | ||||
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetEnrichment; | ||||
| use FireflyIII\Transformers\BudgetTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -55,18 +57,20 @@ class UpdateController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/budgets/updateBudget
 | ||||
|      * | ||||
|      * Update a budget. | ||||
|      */ | ||||
|     public function update(UpdateRequest $request, Budget $budget): JsonResponse | ||||
|     { | ||||
|         $data        = $request->getAll(); | ||||
|         $data['fire_webhooks'] ??= true; | ||||
|         $budget      = $this->repository->update($budget, $data); | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new BudgetEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budget      = $enrichment->enrichSingle($budget); | ||||
| 
 | ||||
|         /** @var BudgetTransformer $transformer */ | ||||
|         $transformer = app(BudgetTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -31,6 +31,8 @@ use FireflyIII\Models\Budget; | ||||
| use FireflyIII\Models\BudgetLimit; | ||||
| use FireflyIII\Repositories\Budget\BudgetLimitRepositoryInterface; | ||||
| use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetLimitEnrichment; | ||||
| use FireflyIII\Transformers\BudgetLimitTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -75,6 +77,18 @@ class ShowController extends Controller | ||||
|      */ | ||||
|     public function index(Budget $budget): JsonResponse | ||||
|     { | ||||
|         /** @var User $admin */ | ||||
|         $admin        = auth()->user(); | ||||
|         // enrich budget:
 | ||||
|         $enrichment   = new BudgetEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
| 
 | ||||
|         /** @var Budget $budget */ | ||||
|         $budget       = $enrichment->enrichSingle($budget); | ||||
| 
 | ||||
| 
 | ||||
|         $manager      = $this->getManager(); | ||||
|         $manager->parseIncludes('budget'); | ||||
|         $pageSize     = $this->parameters->get('limit'); | ||||
| @@ -84,6 +98,11 @@ class ShowController extends Controller | ||||
|         $paginator    = new LengthAwarePaginator($budgetLimits, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.budgets.limits.index', [$budget->id]).$this->buildParams()); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         $enrichment   = new BudgetLimitEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budgetLimits = $enrichment->enrich($budgetLimits); | ||||
| 
 | ||||
|         /** @var BudgetLimitTransformer $transformer */ | ||||
|         $transformer  = app(BudgetLimitTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| @@ -113,6 +132,13 @@ class ShowController extends Controller | ||||
|         $paginator    = new LengthAwarePaginator($budgetLimits, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.budget-limits.index').$this->buildParams()); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin        = auth()->user(); | ||||
|         $enrichment   = new BudgetLimitEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budgetLimits = $enrichment->enrich($budgetLimits); | ||||
| 
 | ||||
|         /** @var BudgetLimitTransformer $transformer */ | ||||
|         $transformer  = app(BudgetLimitTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| @@ -137,6 +163,13 @@ class ShowController extends Controller | ||||
|         // continue!
 | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new BudgetLimitEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budgetLimit = $enrichment->enrichSingle($budgetLimit); | ||||
| 
 | ||||
|         /** @var BudgetLimitTransformer $transformer */ | ||||
|         $transformer = app(BudgetLimitTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -28,6 +28,7 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\BudgetLimit\StoreRequest; | ||||
| use FireflyIII\Models\Budget; | ||||
| use FireflyIII\Repositories\Budget\BudgetLimitRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetLimitEnrichment; | ||||
| use FireflyIII\Transformers\BudgetLimitTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -69,12 +70,19 @@ class StoreController extends Controller | ||||
|         $data               = $request->getAll(); | ||||
|         $data['start_date'] = $data['start']; | ||||
|         $data['end_date']   = $data['end']; | ||||
|         $data['notes']      = $data['notes']; | ||||
|         $data['fire_webhooks'] ??= true; | ||||
|         $data['budget_id']  = $budget->id; | ||||
| 
 | ||||
|         $budgetLimit        = $this->blRepository->store($data); | ||||
|         $manager            = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin              = auth()->user(); | ||||
|         $enrichment         = new BudgetLimitEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budgetLimit        = $enrichment->enrichSingle($budgetLimit); | ||||
| 
 | ||||
|         /** @var BudgetLimitTransformer $transformer */ | ||||
|         $transformer        = app(BudgetLimitTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -30,6 +30,7 @@ use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Budget; | ||||
| use FireflyIII\Models\BudgetLimit; | ||||
| use FireflyIII\Repositories\Budget\BudgetLimitRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetLimitEnrichment; | ||||
| use FireflyIII\Transformers\BudgetLimitTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| @@ -76,10 +77,18 @@ class UpdateController extends Controller | ||||
|             throw new FireflyException('20028: The budget limit does not belong to the budget.'); | ||||
|         } | ||||
|         $data              = $request->getAll(); | ||||
|         $data['fire_webhooks'] ??= true; | ||||
|         $data['budget_id'] = $budget->id; | ||||
|         $budgetLimit       = $this->blRepository->update($budgetLimit, $data); | ||||
|         $manager           = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin             = auth()->user(); | ||||
|         $enrichment        = new BudgetLimitEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budgetLimit       = $enrichment->enrichSingle($budgetLimit); | ||||
| 
 | ||||
|         /** @var BudgetLimitTransformer $transformer */ | ||||
|         $transformer       = app(BudgetLimitTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Category; | ||||
| use FireflyIII\Repositories\Category\CategoryRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\CategoryEnrichment; | ||||
| use FireflyIII\Transformers\CategoryTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| @@ -78,6 +80,15 @@ class ShowController extends Controller | ||||
|         $count       = $collection->count(); | ||||
|         $categories  = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new CategoryEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $categories  = $enrichment->enrich($categories); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($categories, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.categories.index').$this->buildParams()); | ||||
| @@ -105,6 +116,15 @@ class ShowController extends Controller | ||||
|         $transformer = app(CategoryTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new CategoryEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $category    = $enrichment->enrichSingle($category); | ||||
| 
 | ||||
|         $resource    = new Item($category, $transformer, 'categories'); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\Category\StoreRequest; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Repositories\Category\CategoryRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\CategoryEnrichment; | ||||
| use FireflyIII\Transformers\CategoryTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -72,6 +74,15 @@ class StoreController extends Controller | ||||
|         $transformer = app(CategoryTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new CategoryEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $category    = $enrichment->enrichSingle($category); | ||||
| 
 | ||||
|         $resource    = new Item($category, $transformer, 'categories'); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\Category\UpdateRequest; | ||||
| use FireflyIII\Models\Category; | ||||
| use FireflyIII\Repositories\Category\CategoryRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\CategoryEnrichment; | ||||
| use FireflyIII\Transformers\CategoryTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -71,6 +73,15 @@ class UpdateController extends Controller | ||||
|         $transformer = app(CategoryTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new CategoryEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $category    = $enrichment->enrichSingle($category); | ||||
| 
 | ||||
|         $resource    = new Item($category, $transformer, 'categories'); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|   | ||||
| @@ -28,13 +28,11 @@ use Carbon\Carbon; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate\DestroyRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\ValidationException; | ||||
| use FireflyIII\Models\CurrencyExchangeRate; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; | ||||
| 
 | ||||
| class DestroyController extends Controller | ||||
| { | ||||
| @@ -59,23 +57,25 @@ class DestroyController extends Controller | ||||
| 
 | ||||
|     public function destroy(DestroyRequest $request, TransactionCurrency $from, TransactionCurrency $to): JsonResponse | ||||
|     { | ||||
|         $date = $request->getDate(); | ||||
|         if (!$date instanceof Carbon) { | ||||
|             throw new ValidationException('Date is required'); | ||||
|         } | ||||
|         $rate = $this->repository->getSpecificRateOnDate($from, $to, $date); | ||||
|         if (!$rate instanceof CurrencyExchangeRate) { | ||||
|             throw new NotFoundHttpException(); | ||||
|         } | ||||
|         $this->repository->deleteRate($rate); | ||||
|         $this->repository->deleteRates($from, $to); | ||||
| 
 | ||||
|         return response()->json([], 204); | ||||
|     } | ||||
| 
 | ||||
|     public function destroySingle(CurrencyExchangeRate $exchangeRate): JsonResponse | ||||
|     public function destroySingleById(CurrencyExchangeRate $exchangeRate): JsonResponse | ||||
|     { | ||||
|         $this->repository->deleteRate($exchangeRate); | ||||
| 
 | ||||
|         return response()->json([], 204); | ||||
|     } | ||||
| 
 | ||||
|     public function destroySingleByDate(TransactionCurrency $from, TransactionCurrency $to, Carbon $date): JsonResponse | ||||
|     { | ||||
|         $exchangeRate = $this->repository->getSpecificRateOnDate($from, $to, $date); | ||||
|         if ($exchangeRate instanceof CurrencyExchangeRate) { | ||||
|             $this->repository->deleteRate($exchangeRate); | ||||
|         } | ||||
| 
 | ||||
|         return response()->json([], 204); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -24,7 +24,8 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Repositories\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| use FireflyIII\Transformers\ExchangeRateTransformer; | ||||
| @@ -39,7 +40,7 @@ class IndexController extends Controller | ||||
|     use ValidatesUserGroupTrait; | ||||
| 
 | ||||
|     public const string RESOURCE_KEY = 'currency_exchange_rates'; | ||||
| 
 | ||||
|     protected array $acceptedRoles   = [UserRoleEnum::OWNER]; | ||||
|     private ExchangeRateRepositoryInterface $repository; | ||||
| 
 | ||||
|     public function __construct() | ||||
|   | ||||
| @@ -24,7 +24,9 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\CurrencyExchangeRate; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| @@ -32,6 +34,7 @@ use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| use FireflyIII\Transformers\ExchangeRateTransformer; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; | ||||
| 
 | ||||
| /** | ||||
|  * Class ShowController | ||||
| @@ -41,7 +44,7 @@ class ShowController extends Controller | ||||
|     use ValidatesUserGroupTrait; | ||||
| 
 | ||||
|     public const string RESOURCE_KEY = 'exchange-rates'; | ||||
| 
 | ||||
|     protected array $acceptedRoles   = [UserRoleEnum::OWNER]; | ||||
|     private ExchangeRateRepositoryInterface $repository; | ||||
| 
 | ||||
|     public function __construct() | ||||
| @@ -75,7 +78,7 @@ class ShowController extends Controller | ||||
|         ; | ||||
|     } | ||||
| 
 | ||||
|     public function showSingle(CurrencyExchangeRate $exchangeRate): JsonResponse | ||||
|     public function showSingleById(CurrencyExchangeRate $exchangeRate): JsonResponse | ||||
|     { | ||||
|         $transformer = new ExchangeRateTransformer(); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| @@ -85,4 +88,20 @@ class ShowController extends Controller | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| 
 | ||||
|     public function showSingleByDate(TransactionCurrency $from, TransactionCurrency $to, Carbon $date): JsonResponse | ||||
|     { | ||||
|         $transformer  = new ExchangeRateTransformer(); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $exchangeRate = $this->repository->getSpecificRateOnDate($from, $to, $date); | ||||
|         if (!$exchangeRate instanceof CurrencyExchangeRate) { | ||||
|             throw new NotFoundHttpException(); | ||||
|         } | ||||
| 
 | ||||
|         return response() | ||||
|             ->api($this->jsonApiObject(self::RESOURCE_KEY, $exchangeRate, $transformer)) | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -24,20 +24,28 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use FireflyIII\Models\CurrencyExchangeRate; | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate\StoreByCurrenciesRequest; | ||||
| use FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate\StoreByDateRequest; | ||||
| use FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate\StoreRequest; | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\CurrencyExchangeRate; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| use FireflyIII\Transformers\ExchangeRateTransformer; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use Illuminate\Support\Collection; | ||||
| 
 | ||||
| class StoreController extends Controller | ||||
| { | ||||
|     use ValidatesUserGroupTrait; | ||||
| 
 | ||||
|     public const string RESOURCE_KEY = 'exchange-rates'; | ||||
| 
 | ||||
|     public const string RESOURCE_KEY                       = 'exchange-rates'; | ||||
|     protected array                         $acceptedRoles = [UserRoleEnum::OWNER]; | ||||
|     private ExchangeRateRepositoryInterface $repository; | ||||
| 
 | ||||
|     public function __construct() | ||||
| @@ -53,6 +61,68 @@ class StoreController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     public function storeByCurrencies(StoreByCurrenciesRequest $request, TransactionCurrency $from, TransactionCurrency $to): JsonResponse | ||||
|     { | ||||
| 
 | ||||
|         $data        = $request->getAll(); | ||||
|         $collection  = new Collection(); | ||||
| 
 | ||||
|         foreach ($data as $date => $rate) { | ||||
|             $date     = Carbon::createFromFormat('Y-m-d', $date); | ||||
|             $existing = $this->repository->getSpecificRateOnDate($from, $to, $date); | ||||
|             if ($existing instanceof CurrencyExchangeRate) { | ||||
|                 // update existing rate.
 | ||||
|                 $existing = $this->repository->updateExchangeRate($existing, $rate); | ||||
|                 $collection->push($existing); | ||||
| 
 | ||||
|                 continue; | ||||
|             } | ||||
|             $new      = $this->repository->storeExchangeRate($from, $to, $rate, $date); | ||||
|             $collection->push($new); | ||||
|         } | ||||
| 
 | ||||
|         $count       = $collection->count(); | ||||
|         $paginator   = new LengthAwarePaginator($collection, $count, $count, 1); | ||||
|         $transformer = new ExchangeRateTransformer(); | ||||
|         $transformer->setParameters($this->parameters); // give params to transformer
 | ||||
| 
 | ||||
|         return response() | ||||
|             ->json($this->jsonApiList(self::RESOURCE_KEY, $paginator, $transformer)) | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| 
 | ||||
|     public function storeByDate(StoreByDateRequest $request, Carbon $date): JsonResponse | ||||
|     { | ||||
| 
 | ||||
|         $data        = $request->getAll(); | ||||
|         $from        = $request->getFromCurrency(); | ||||
|         $collection  = new Collection(); | ||||
|         foreach ($data['rates'] as $key => $rate) { | ||||
|             $to       = Amount::getTransactionCurrencyByCode($key); | ||||
|             $existing = $this->repository->getSpecificRateOnDate($from, $to, $date); | ||||
|             if ($existing instanceof CurrencyExchangeRate) { | ||||
|                 // update existing rate.
 | ||||
|                 $existing = $this->repository->updateExchangeRate($existing, $rate); | ||||
|                 $collection->push($existing); | ||||
| 
 | ||||
|                 continue; | ||||
|             } | ||||
|             $new      = $this->repository->storeExchangeRate($from, $to, $rate, $date); | ||||
|             $collection->push($new); | ||||
|         } | ||||
| 
 | ||||
|         $count       = $collection->count(); | ||||
|         $paginator   = new LengthAwarePaginator($collection, $count, $count, 1); | ||||
|         $transformer = new ExchangeRateTransformer(); | ||||
|         $transformer->setParameters($this->parameters); // give params to transformer
 | ||||
| 
 | ||||
|         return response() | ||||
|             ->json($this->jsonApiList(self::RESOURCE_KEY, $paginator, $transformer)) | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| 
 | ||||
|     public function store(StoreRequest $request): JsonResponse | ||||
|     { | ||||
|         $date        = $request->getDate(); | ||||
|   | ||||
| @@ -24,20 +24,24 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate\UpdateRequest; | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Models\CurrencyExchangeRate; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| use FireflyIII\Transformers\ExchangeRateTransformer; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Symfony\Component\HttpKernel\Exception\NotFoundHttpException; | ||||
| 
 | ||||
| class UpdateController extends Controller | ||||
| { | ||||
|     use ValidatesUserGroupTrait; | ||||
| 
 | ||||
|     public const string RESOURCE_KEY = 'exchange-rates'; | ||||
| 
 | ||||
|     public const string RESOURCE_KEY                       = 'exchange-rates'; | ||||
|     protected array                         $acceptedRoles = [UserRoleEnum::OWNER]; | ||||
|     private ExchangeRateRepositoryInterface $repository; | ||||
| 
 | ||||
|     public function __construct() | ||||
| @@ -53,7 +57,7 @@ class UpdateController extends Controller | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     public function update(UpdateRequest $request, CurrencyExchangeRate $exchangeRate): JsonResponse | ||||
|     public function updateById(UpdateRequest $request, CurrencyExchangeRate $exchangeRate): JsonResponse | ||||
|     { | ||||
|         $date         = $request->getDate(); | ||||
|         $rate         = $request->getRate(); | ||||
| @@ -66,4 +70,23 @@ class UpdateController extends Controller | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| 
 | ||||
|     public function updateByDate(UpdateRequest $request, TransactionCurrency $from, TransactionCurrency $to, Carbon $date): JsonResponse | ||||
|     { | ||||
|         $exchangeRate = $this->repository->getSpecificRateOnDate($from, $to, $date); | ||||
|         if (!$exchangeRate instanceof CurrencyExchangeRate) { | ||||
|             throw new NotFoundHttpException(); | ||||
|         } | ||||
|         $date         = $request->getDate(); | ||||
|         $rate         = $request->getRate(); | ||||
|         $exchangeRate = $this->repository->updateExchangeRate($exchangeRate, $rate, $date); | ||||
| 
 | ||||
|         $transformer  = new ExchangeRateTransformer(); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         return response() | ||||
|             ->api($this->jsonApiObject(self::RESOURCE_KEY, $exchangeRate, $transformer)) | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -28,6 +28,8 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\ObjectGroup; | ||||
| use FireflyIII\Repositories\ObjectGroup\ObjectGroupRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\PiggyBankEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\SubscriptionEnrichment; | ||||
| use FireflyIII\Transformers\BillTransformer; | ||||
| use FireflyIII\Transformers\PiggyBankTransformer; | ||||
| use FireflyIII\User; | ||||
| @@ -79,6 +81,15 @@ class ListController extends Controller | ||||
|         $count       = $collection->count(); | ||||
|         $bills       = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new SubscriptionEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $bills       = $enrichment->enrich($bills); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($bills, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.currencies.bills', [$objectGroup->id]).$this->buildParams()); | ||||
| @@ -114,6 +125,13 @@ class ListController extends Controller | ||||
|         $count       = $collection->count(); | ||||
|         $piggyBanks  = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new PiggyBankEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $piggyBanks  = $enrichment->enrich($piggyBanks); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($piggyBanks, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.object-groups.piggy-banks', [$objectGroup->id]).$this->buildParams()); | ||||
| @@ -122,7 +140,7 @@ class ListController extends Controller | ||||
|         $transformer = app(PiggyBankTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new FractalCollection($piggyBanks, $transformer, 'piggy_banks'); | ||||
|         $resource    = new FractalCollection($piggyBanks, $transformer, 'piggy-banks'); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|   | ||||
| @@ -29,6 +29,7 @@ use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\PiggyBank; | ||||
| use FireflyIII\Repositories\PiggyBank\PiggyBankRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\AccountEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\PiggyBankEventEnrichment; | ||||
| use FireflyIII\Transformers\AccountTransformer; | ||||
| use FireflyIII\Transformers\AttachmentTransformer; | ||||
| use FireflyIII\Transformers\PiggyBankEventTransformer; | ||||
| @@ -83,8 +84,8 @@ class ListController extends Controller | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new AccountEnrichment(); | ||||
|         $enrichment->setDate($this->parameters->get('date')); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setNative($this->nativeCurrency); | ||||
|         $accounts    = $enrichment->enrich($accounts); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
| @@ -148,6 +149,13 @@ class ListController extends Controller | ||||
|         $count       = $collection->count(); | ||||
|         $events      = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new PiggyBankEventEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $events      = $enrichment->enrich($events); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($events, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.piggy-banks.events', [$piggyBank->id]).$this->buildParams()); | ||||
| @@ -156,7 +164,7 @@ class ListController extends Controller | ||||
|         $transformer = app(PiggyBankEventTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new FractalCollection($events, $transformer, 'piggy_bank_events'); | ||||
|         $resource    = new FractalCollection($events, $transformer, sprintf('piggy-banks/%d/events', $piggyBank->id)); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\PiggyBank; | ||||
| use FireflyIII\Repositories\PiggyBank\PiggyBankRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\PiggyBankEnrichment; | ||||
| use FireflyIII\Transformers\PiggyBankTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| @@ -77,6 +79,13 @@ class ShowController extends Controller | ||||
|         $count       = $collection->count(); | ||||
|         $piggyBanks  = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new PiggyBankEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $piggyBanks  = $enrichment->enrich($piggyBanks); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($piggyBanks, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.piggy-banks.index').$this->buildParams()); | ||||
| @@ -85,7 +94,7 @@ class ShowController extends Controller | ||||
|         $transformer = app(PiggyBankTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new FractalCollection($piggyBanks, $transformer, 'piggy_banks'); | ||||
|         $resource    = new FractalCollection($piggyBanks, $transformer, 'piggy-banks'); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
| @@ -101,11 +110,19 @@ class ShowController extends Controller | ||||
|     { | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new PiggyBankEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $piggyBank   = $enrichment->enrichSingle($piggyBank); | ||||
| 
 | ||||
| 
 | ||||
|         /** @var PiggyBankTransformer $transformer */ | ||||
|         $transformer = app(PiggyBankTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new Item($piggyBank, $transformer, 'piggy_banks'); | ||||
|         $resource    = new Item($piggyBank, $transformer, 'piggy-banks'); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|     } | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\PiggyBank\StoreRequest; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Repositories\PiggyBank\PiggyBankRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\PiggyBankEnrichment; | ||||
| use FireflyIII\Transformers\PiggyBankTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -68,6 +70,13 @@ class StoreController extends Controller | ||||
|         $piggyBank   = $this->repository->store($request->getAll()); | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new PiggyBankEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $piggyBank   = $enrichment->enrichSingle($piggyBank); | ||||
| 
 | ||||
|         /** @var PiggyBankTransformer $transformer */ | ||||
|         $transformer = app(PiggyBankTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\PiggyBank\UpdateRequest; | ||||
| use FireflyIII\Models\PiggyBank; | ||||
| use FireflyIII\Repositories\PiggyBank\PiggyBankRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\PiggyBankEnrichment; | ||||
| use FireflyIII\Transformers\PiggyBankTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -66,9 +68,12 @@ class UpdateController extends Controller | ||||
|         $data        = $request->getAll(); | ||||
|         $piggyBank   = $this->repository->update($piggyBank, $data); | ||||
| 
 | ||||
|         if (array_key_exists('current_amount', $data) && '' !== $data['current_amount']) { | ||||
|             $this->repository->setCurrentAmount($piggyBank, $data['current_amount']); | ||||
|         } | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new PiggyBankEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $piggyBank   = $enrichment->enrichSingle($piggyBank); | ||||
| 
 | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
| @@ -76,7 +81,7 @@ class UpdateController extends Controller | ||||
|         $transformer = app(PiggyBankTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new Item($piggyBank, $transformer, 'piggy_banks'); | ||||
|         $resource    = new Item($piggyBank, $transformer, 'piggy-banks'); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|     } | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\Recurrence; | ||||
| use FireflyIII\Repositories\Recurring\RecurringRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\RecurringEnrichment; | ||||
| use FireflyIII\Transformers\RecurrenceTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| @@ -76,17 +78,24 @@ class ShowController extends Controller | ||||
|         // get list of budgets. Count it and split it.
 | ||||
|         $collection  = $this->repository->get(); | ||||
|         $count       = $collection->count(); | ||||
|         $piggyBanks  = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
|         $recurrences = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new RecurringEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $recurrences = $enrichment->enrich($recurrences); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($piggyBanks, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator   = new LengthAwarePaginator($recurrences, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.recurrences.index').$this->buildParams()); | ||||
| 
 | ||||
|         /** @var RecurrenceTransformer $transformer */ | ||||
|         $transformer = app(RecurrenceTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new FractalCollection($piggyBanks, $transformer, 'recurrences'); | ||||
|         $resource    = new FractalCollection($recurrences, $transformer, 'recurrences'); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
| @@ -102,6 +111,13 @@ class ShowController extends Controller | ||||
|     { | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new RecurringEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $recurrence  = $enrichment->enrichSingle($recurrence); | ||||
| 
 | ||||
|         /** @var RecurrenceTransformer $transformer */ | ||||
|         $transformer = app(RecurrenceTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\Recurrence\StoreRequest; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Repositories\Recurring\RecurringRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\RecurringEnrichment; | ||||
| use FireflyIII\Transformers\RecurrenceTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -69,6 +71,13 @@ class StoreController extends Controller | ||||
|         $recurrence  = $this->repository->store($data); | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new RecurringEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $recurrence  = $enrichment->enrichSingle($recurrence); | ||||
| 
 | ||||
|         /** @var RecurrenceTransformer $transformer */ | ||||
|         $transformer = app(RecurrenceTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
							
								
								
									
										126
									
								
								app/Api/V1/Controllers/Models/Recurrence/TriggerController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										126
									
								
								app/Api/V1/Controllers/Models/Recurrence/TriggerController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,126 @@ | ||||
| <?php | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| /* | ||||
|  * TriggerController.php | ||||
|  * Copyright (c) 2025 james@firefly-iii.org | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU Affero General Public License as | ||||
|  * published by the Free Software Foundation, either version 3 of the | ||||
|  * License, or (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU Affero General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU Affero General Public License | ||||
|  * along with this program.  If not, see <https://www.gnu.org/licenses/>. | ||||
|  */ | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\Recurrence; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Generic\SingleDateRequest; | ||||
| use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Jobs\CreateRecurringTransactions; | ||||
| use FireflyIII\Models\Recurrence; | ||||
| use FireflyIII\Repositories\Recurring\RecurringRepositoryInterface; | ||||
| use FireflyIII\Support\Facades\Preferences; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\TransactionGroupEnrichment; | ||||
| use FireflyIII\Transformers\TransactionGroupTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use Illuminate\Support\Collection; | ||||
| use Illuminate\Support\Facades\Log; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| use League\Fractal\Resource\Collection as FractalCollection; | ||||
| 
 | ||||
| class TriggerController extends Controller | ||||
| { | ||||
|     private RecurringRepositoryInterface $repository; | ||||
| 
 | ||||
|     /** | ||||
|      * RecurrenceController constructor. | ||||
|      */ | ||||
|     public function __construct() | ||||
|     { | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->repository = app(RecurringRepositoryInterface::class); | ||||
|                 $this->repository->setUser(auth()->user()); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     public function trigger(SingleDateRequest $request, Recurrence $recurrence): JsonResponse | ||||
|     { | ||||
|         // find recurrence occurrence for this date and trigger it.
 | ||||
|         // grab the date from the last time the recurrence fired:
 | ||||
|         $backupDate                 = $recurrence->latest_date; | ||||
|         $date                       = $request->getDate(); | ||||
| 
 | ||||
|         // fire the recurring cron job on the given date, then post-date the created transaction.
 | ||||
|         Log::info(sprintf('Trigger: will now fire recurring cron job task for date "%s".', $date->format('Y-m-d H:i:s'))); | ||||
| 
 | ||||
|         /** @var CreateRecurringTransactions $job */ | ||||
|         $job                        = app(CreateRecurringTransactions::class); | ||||
|         $job->setRecurrences(new Collection()->push($recurrence)); | ||||
|         $job->setDate($date); | ||||
|         $job->setForce(false); | ||||
|         $job->handle(); | ||||
|         Log::debug('Done with recurrence.'); | ||||
| 
 | ||||
|         $groups                     = $job->getGroups(); | ||||
|         $this->repository->markGroupsAsNow($groups); | ||||
|         $recurrence->latest_date    = $backupDate; | ||||
|         $recurrence->latest_date_tz = $backupDate?->format('e'); | ||||
|         $recurrence->save(); | ||||
|         Preferences::mark(); | ||||
| 
 | ||||
|         // enrich groups and return them:
 | ||||
| 
 | ||||
|         if (0 === $groups->count()) { | ||||
|             $paginator = new LengthAwarePaginator(new Collection(), 0, 1); | ||||
|         } | ||||
|         if ($groups->count() > 0) { | ||||
|             /** @var User $admin */ | ||||
|             $admin     = auth()->user(); | ||||
| 
 | ||||
|             // use new group collector:
 | ||||
|             /** @var GroupCollectorInterface $collector */ | ||||
|             $collector = app(GroupCollectorInterface::class); | ||||
|             $collector | ||||
|                 ->setUser($admin) | ||||
|                 ->setIds($groups->pluck('id')->toArray()) | ||||
|                 ->withAPIInformation() | ||||
|             ; | ||||
|             $paginator = $collector->getPaginatedGroups(); | ||||
|         } | ||||
| 
 | ||||
|         $manager                    = $this->getManager(); | ||||
|         $paginator->setPath(route('api.v1.recurrences.trigger', [$recurrence->id]).$this->buildParams()); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         $admin                      = auth()->user(); | ||||
|         $enrichment                 = new TransactionGroupEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $transactions               = $enrichment->enrich($paginator->getCollection()); | ||||
| 
 | ||||
|         /** @var TransactionGroupTransformer $transformer */ | ||||
|         $transformer                = app(TransactionGroupTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource                   = new FractalCollection($transactions, $transformer, 'transactions'); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|     } | ||||
| } | ||||
| @@ -28,7 +28,9 @@ use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\Recurrence\UpdateRequest; | ||||
| use FireflyIII\Models\Recurrence; | ||||
| use FireflyIII\Repositories\Recurring\RecurringRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\RecurringEnrichment; | ||||
| use FireflyIII\Transformers\RecurrenceTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use League\Fractal\Resource\Item; | ||||
| 
 | ||||
| @@ -67,6 +69,13 @@ class UpdateController extends Controller | ||||
|         $recurrence  = $this->repository->update($recurrence, $data); | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new RecurringEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $recurrence  = $enrichment->enrichSingle($recurrence); | ||||
| 
 | ||||
|         /** @var RecurrenceTransformer $transformer */ | ||||
|         $transformer = app(RecurrenceTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -75,7 +75,7 @@ class TriggerController extends Controller | ||||
| 
 | ||||
|         /** @var RuleEngineInterface $ruleEngine */ | ||||
|         $ruleEngine   = app(RuleEngineInterface::class); | ||||
|         $ruleEngine->setRules(new Collection([$rule])); | ||||
|         $ruleEngine->setRules(new Collection()->push($rule)); | ||||
| 
 | ||||
|         // overrule the rule(s) if necessary.
 | ||||
|         if (array_key_exists('start', $parameters) && null !== $parameters['start']) { | ||||
| @@ -129,7 +129,7 @@ class TriggerController extends Controller | ||||
| 
 | ||||
|         /** @var RuleEngineInterface $ruleEngine */ | ||||
|         $ruleEngine = app(RuleEngineInterface::class); | ||||
|         $ruleEngine->setRules(new Collection([$rule])); | ||||
|         $ruleEngine->setRules(new Collection()->push($rule)); | ||||
| 
 | ||||
|         // overrule the rule(s) if necessary.
 | ||||
|         if (array_key_exists('start', $parameters) && null !== $parameters['start']) { | ||||
|   | ||||
| @@ -29,6 +29,7 @@ use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Models\TransactionGroup; | ||||
| use FireflyIII\Models\TransactionJournal; | ||||
| use FireflyIII\Repositories\Journal\JournalAPIRepositoryInterface; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\PiggyBankEventEnrichment; | ||||
| use FireflyIII\Transformers\AttachmentTransformer; | ||||
| use FireflyIII\Transformers\PiggyBankEventTransformer; | ||||
| use FireflyIII\Transformers\TransactionLinkTransformer; | ||||
| @@ -113,6 +114,14 @@ class ListController extends Controller | ||||
|         } | ||||
|         $count       = $collection->count(); | ||||
|         $events      = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new PiggyBankEventEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $events      = $enrichment->enrich($events); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($events, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.transactions.piggy-bank-events', [$transactionGroup->id]).$this->buildParams()); | ||||
|   | ||||
| @@ -147,6 +147,7 @@ class ShowController extends Controller | ||||
|         $enrichment->setUser($admin); | ||||
|         $selectedGroup = $enrichment->enrichSingle($selectedGroup); | ||||
| 
 | ||||
| 
 | ||||
|         /** @var TransactionGroupTransformer $transformer */ | ||||
|         $transformer   = app(TransactionGroupTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
|   | ||||
| @@ -32,6 +32,7 @@ use FireflyIII\Repositories\Currency\CurrencyRepositoryInterface; | ||||
| use FireflyIII\Repositories\User\UserRepositoryInterface; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Validation\ValidationException; | ||||
| 
 | ||||
| /** | ||||
|  * Class DestroyController | ||||
| @@ -65,6 +66,7 @@ class DestroyController extends Controller | ||||
|      * Remove the specified resource from storage. | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      * @throws ValidationException | ||||
|      */ | ||||
|     public function destroy(TransactionCurrency $currency): JsonResponse | ||||
|     { | ||||
|   | ||||
| @@ -30,9 +30,7 @@ use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Models\Account; | ||||
| use FireflyIII\Models\Bill; | ||||
| use FireflyIII\Models\Recurrence; | ||||
| use FireflyIII\Models\RecurrenceTransaction; | ||||
| use FireflyIII\Models\Rule; | ||||
| use FireflyIII\Models\RuleTrigger; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Repositories\Bill\BillRepositoryInterface; | ||||
| @@ -43,6 +41,9 @@ use FireflyIII\Repositories\Rule\RuleRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\AccountFilter; | ||||
| use FireflyIII\Support\Http\Api\TransactionFilter; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\AccountEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\BudgetLimitEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\RecurringEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\SubscriptionEnrichment; | ||||
| use FireflyIII\Support\JsonApi\Enrichments\TransactionGroupEnrichment; | ||||
| use FireflyIII\Transformers\AccountTransformer; | ||||
| use FireflyIII\Transformers\AvailableBudgetTransformer; | ||||
| @@ -106,8 +107,8 @@ class ListController extends Controller | ||||
|         /** @var User $admin */ | ||||
|         $admin             = auth()->user(); | ||||
|         $enrichment        = new AccountEnrichment(); | ||||
|         $enrichment->setDate($this->parameters->get('date')); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setNative($this->nativeCurrency); | ||||
|         $accounts          = $enrichment->enrich($accounts); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
| @@ -182,6 +183,15 @@ class ListController extends Controller | ||||
|         $count       = $collection->count(); | ||||
|         $bills       = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin       = auth()->user(); | ||||
|         $enrichment  = new SubscriptionEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $enrichment->setStart($this->parameters->get('start')); | ||||
|         $enrichment->setEnd($this->parameters->get('end')); | ||||
|         $bills       = $enrichment->enrich($bills); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($bills, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.currencies.bills', [$currency->code]).$this->buildParams()); | ||||
| @@ -217,6 +227,13 @@ class ListController extends Controller | ||||
|         $paginator    = new LengthAwarePaginator($budgetLimits, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.currencies.budget-limits', [$currency->code]).$this->buildParams()); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin        = auth()->user(); | ||||
|         $enrichment   = new BudgetLimitEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $budgetLimits = $enrichment->enrich($budgetLimits); | ||||
| 
 | ||||
|         /** @var BudgetLimitTransformer $transformer */ | ||||
|         $transformer  = app(BudgetLimitTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| @@ -249,28 +266,32 @@ class ListController extends Controller | ||||
|         // filter selection
 | ||||
|         $collection     = $unfiltered->filter( | ||||
|             static function (Recurrence $recurrence) use ($currency) {  // @phpstan-ignore-line
 | ||||
|                 /** @var RecurrenceTransaction $transaction */ | ||||
|                 foreach ($recurrence->recurrenceTransactions as $transaction) { | ||||
|                     if ($transaction->transaction_currency_id === $currency->id || $transaction->foreign_currency_id === $currency->id) { | ||||
|                         return $recurrence; | ||||
|                     } | ||||
|                 if (array_any($recurrence->recurrenceTransactions, fn ($transaction) => $transaction->transaction_currency_id === $currency->id || $transaction->foreign_currency_id === $currency->id)) { | ||||
|                     return $recurrence; | ||||
|                 } | ||||
| 
 | ||||
|                 return null; | ||||
|             } | ||||
|         ); | ||||
|         $count          = $collection->count(); | ||||
|         $piggyBanks     = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
|         $recurrences    = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // enrich
 | ||||
|         /** @var User $admin */ | ||||
|         $admin          = auth()->user(); | ||||
|         $enrichment     = new RecurringEnrichment(); | ||||
|         $enrichment->setUser($admin); | ||||
|         $recurrences    = $enrichment->enrich($recurrences); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator      = new LengthAwarePaginator($piggyBanks, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator      = new LengthAwarePaginator($recurrences, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.currencies.recurrences', [$currency->code]).$this->buildParams()); | ||||
| 
 | ||||
|         /** @var RecurrenceTransformer $transformer */ | ||||
|         $transformer    = app(RecurrenceTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource       = new FractalCollection($piggyBanks, $transformer, 'recurrences'); | ||||
|         $resource       = new FractalCollection($recurrences, $transformer, 'recurrences'); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
| @@ -296,11 +317,8 @@ class ListController extends Controller | ||||
| 
 | ||||
|         $collection  = $unfiltered->filter( | ||||
|             static function (Rule $rule) use ($currency) { // @phpstan-ignore-line
 | ||||
|                 /** @var RuleTrigger $trigger */ | ||||
|                 foreach ($rule->ruleTriggers as $trigger) { | ||||
|                     if ('currency_is' === $trigger->trigger_type && $currency->name === $trigger->trigger_value) { | ||||
|                         return $rule; | ||||
|                     } | ||||
|                 if (array_any($rule->ruleTriggers, fn ($trigger) => 'currency_is' === $trigger->trigger_type && $currency->name === $trigger->trigger_value)) { | ||||
|                     return $rule; | ||||
|                 } | ||||
| 
 | ||||
|                 return null; | ||||
|   | ||||
| @@ -107,7 +107,7 @@ class ShowController extends Controller | ||||
|         /** @var User $user */ | ||||
|         $user        = auth()->user(); | ||||
|         $manager     = $this->getManager(); | ||||
|         $this->parameters->set('nativeCurrency', $this->nativeCurrency); | ||||
|         $this->parameters->set('primaryCurrency', $this->primaryCurrency); | ||||
| 
 | ||||
|         // update fields with user info.
 | ||||
|         $currency->refreshForUser($user); | ||||
| @@ -122,19 +122,16 @@ class ShowController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/currencies/getNativeCurrency
 | ||||
|      * | ||||
|      * Show a currency. | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function showDefault(): JsonResponse | ||||
|     public function showPrimary(): JsonResponse | ||||
|     { | ||||
|         /** @var User $user */ | ||||
|         $user        = auth()->user(); | ||||
|         $manager     = $this->getManager(); | ||||
|         $currency    = $this->nativeCurrency; | ||||
|         $currency    = $this->primaryCurrency; | ||||
| 
 | ||||
|         // update fields with user info.
 | ||||
|         $currency->refreshForUser($user); | ||||
|   | ||||
| @@ -73,7 +73,7 @@ class StoreController extends Controller | ||||
|     { | ||||
|         $currency    = $this->repository->store($request->getAll()); | ||||
|         if (true === $request->boolean('default')) { | ||||
|             $this->repository->makeDefault($currency); | ||||
|             $this->repository->makePrimary($currency); | ||||
|             app('preferences')->mark(); | ||||
|         } | ||||
|         $manager     = $this->getManager(); | ||||
|   | ||||
| @@ -99,19 +99,14 @@ class UpdateController extends Controller | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/currencies/nativeCurrency
 | ||||
|      * | ||||
|      * Make the currency a default currency. | ||||
|      * | ||||
|      * @throws FireflyException | ||||
|      */ | ||||
|     public function makeDefault(TransactionCurrency $currency): JsonResponse | ||||
|     public function makePrimary(TransactionCurrency $currency): JsonResponse | ||||
|     { | ||||
|         /** @var User $user */ | ||||
|         $user        = auth()->user(); | ||||
|         $this->repository->enable($currency); | ||||
|         $this->repository->makeDefault($currency); | ||||
|         $this->repository->makePrimary($currency); | ||||
| 
 | ||||
|         app('preferences')->mark(); | ||||
| 
 | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user