mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2026-01-08 03:21:20 +00:00
Compare commits
178 Commits
develop-20
...
develop-20
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e8618047bd | ||
|
|
f104b76f73 | ||
|
|
cb701d8506 | ||
|
|
70a334c56e | ||
|
|
e6b2db1e29 | ||
|
|
e8dffa0052 | ||
|
|
c4f0512f39 | ||
|
|
3268019d0c | ||
|
|
a0ef6a1fc8 | ||
|
|
99d0098b20 | ||
|
|
a7a54c042c | ||
|
|
c44e48a793 | ||
|
|
53b501ca73 | ||
|
|
322f70bcca | ||
|
|
35559c077b | ||
|
|
590ffe7c76 | ||
|
|
8a2d8f148e | ||
|
|
4f0e15e07d | ||
|
|
7463861e0c | ||
|
|
1e70fa28be | ||
|
|
26c6ca470b | ||
|
|
5e54034e0e | ||
|
|
25873ef734 | ||
|
|
1092b04b22 | ||
|
|
01ce74dd72 | ||
|
|
41430d8386 | ||
|
|
01eb19169c | ||
|
|
cfaa7d7c68 | ||
|
|
14d3312a10 | ||
|
|
87be478dd8 | ||
|
|
0b6877a20e | ||
|
|
7186f0ef60 | ||
|
|
538933691e | ||
|
|
46c49ddbd8 | ||
|
|
bcfb134b6e | ||
|
|
57981f1cf9 | ||
|
|
0310186fb7 | ||
|
|
4dcb38290e | ||
|
|
2f5c37048b | ||
|
|
370c8b16ae | ||
|
|
af0555592a | ||
|
|
9c07ddaed6 | ||
|
|
bb7355a566 | ||
|
|
1d48347f8c | ||
|
|
060b76ca9c | ||
|
|
2b2b9b6f7a | ||
|
|
f3dd05a0c0 | ||
|
|
47a91aa273 | ||
|
|
41bc236603 | ||
|
|
65349451ea | ||
|
|
e77b6a55a4 | ||
|
|
2379bcff11 | ||
|
|
7133156fa1 | ||
|
|
a59176689d | ||
|
|
bc2d8f3dfb | ||
|
|
ddf89a9d5a | ||
|
|
7daaba17f6 | ||
|
|
9cb5b1384f | ||
|
|
7d13263482 | ||
|
|
d9ff252915 | ||
|
|
51ba550251 | ||
|
|
fd21c467ad | ||
|
|
9aa90650b4 | ||
|
|
d892257e8b | ||
|
|
db0dbcfcf1 | ||
|
|
f591996f04 | ||
|
|
b08d385586 | ||
|
|
20ef22f67e | ||
|
|
c888baf542 | ||
|
|
8b0af3f666 | ||
|
|
7043e1e7c0 | ||
|
|
c5854eba23 | ||
|
|
ddf1a8cebb | ||
|
|
7dcaf167e9 | ||
|
|
b359d51d3a | ||
|
|
3913fa5086 | ||
|
|
ab2772abe0 | ||
|
|
bc7875b17b | ||
|
|
4938fa9990 | ||
|
|
84df2c80ee | ||
|
|
dc17060754 | ||
|
|
e2fa81dddc | ||
|
|
182dfc95fe | ||
|
|
c8979b6c33 | ||
|
|
ab872e8912 | ||
|
|
d36b94fabf | ||
|
|
e3d4ceaecb | ||
|
|
e3a6e5b788 | ||
|
|
57235c0e00 | ||
|
|
2298c3ddaf | ||
|
|
7224f1be6f | ||
|
|
1bd3019c16 | ||
|
|
f0fa21dead | ||
|
|
845eaed8d7 | ||
|
|
b3649cd4d0 | ||
|
|
55f14c587b | ||
|
|
441a8a8408 | ||
|
|
060c9648f1 | ||
|
|
7680c8733f | ||
|
|
5a0af5c93b | ||
|
|
f4b066add1 | ||
|
|
9ecb414b02 | ||
|
|
ad4f908c24 | ||
|
|
025f739442 | ||
|
|
6df7354c48 | ||
|
|
3f77c845ca | ||
|
|
d4771f7a5c | ||
|
|
ec4e2bfa4f | ||
|
|
dfdbfae4b5 | ||
|
|
349d38b956 | ||
|
|
2267aa3ac4 | ||
|
|
2323aa454e | ||
|
|
8b3317b665 | ||
|
|
15f893c343 | ||
|
|
309b3e765e | ||
|
|
d3fad06e00 | ||
|
|
834f24c99c | ||
|
|
35291e1298 | ||
|
|
ac4e9dcbc5 | ||
|
|
d57806f2ba | ||
|
|
3b005c317d | ||
|
|
e91903fed2 | ||
|
|
fee2002b0f | ||
|
|
f12e502eb8 | ||
|
|
24e62b1cee | ||
|
|
f559ec73e0 | ||
|
|
530b501fcf | ||
|
|
d5ea78025e | ||
|
|
3413b9b5b5 | ||
|
|
0b45c1aa76 | ||
|
|
5718d1690a | ||
|
|
67b16cc070 | ||
|
|
5746ac3247 | ||
|
|
8a2c520b11 | ||
|
|
f46c14df8c | ||
|
|
009fbba491 | ||
|
|
53d84347c2 | ||
|
|
1961487055 | ||
|
|
c9ce5df74b | ||
|
|
1371b6773e | ||
|
|
b9f1baf150 | ||
|
|
66b322e844 | ||
|
|
487b65b669 | ||
|
|
9078781d61 | ||
|
|
1ec830521a | ||
|
|
c4bf2aae7d | ||
|
|
69ca88d9f8 | ||
|
|
b38b7b2534 | ||
|
|
f19bfc3b4b | ||
|
|
d22f9c09d7 | ||
|
|
fc2da9eb42 | ||
|
|
f2c9e20aef | ||
|
|
16b8ca2746 | ||
|
|
46ea074821 | ||
|
|
d2c89781e2 | ||
|
|
e54d711891 | ||
|
|
84d3ad4764 | ||
|
|
b908951a2d | ||
|
|
8b87deea58 | ||
|
|
0d7325b3dc | ||
|
|
a3fd99a498 | ||
|
|
0ff405d1e0 | ||
|
|
46a60af966 | ||
|
|
591c9e3b39 | ||
|
|
c30461b20b | ||
|
|
2c3f86d9bc | ||
|
|
34349e4475 | ||
|
|
6acd5be5dc | ||
|
|
55a2b4e789 | ||
|
|
f41397eb43 | ||
|
|
41fc1e8f82 | ||
|
|
bee219ebf7 | ||
|
|
438f602961 | ||
|
|
429e72e681 | ||
|
|
7a134781f2 | ||
|
|
b572c1dcd3 | ||
|
|
95593f847b | ||
|
|
daddee7806 |
@@ -22,6 +22,9 @@
|
|||||||
|
|
||||||
SCRIPT_DIR="$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )"
|
SCRIPT_DIR="$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd )"
|
||||||
|
|
||||||
|
echo "Running PHP CS Fixer"
|
||||||
$SCRIPT_DIR/phpcs.sh
|
$SCRIPT_DIR/phpcs.sh
|
||||||
|
echo "Running PHPStan"
|
||||||
$SCRIPT_DIR/phpstan.sh
|
$SCRIPT_DIR/phpstan.sh
|
||||||
|
echo "Running PHPMD"
|
||||||
$SCRIPT_DIR/phpmd.sh
|
$SCRIPT_DIR/phpmd.sh
|
||||||
|
|||||||
40
.ci/php-cs-fixer/composer.lock
generated
40
.ci/php-cs-fixer/composer.lock
generated
@@ -8,16 +8,16 @@
|
|||||||
"packages": [
|
"packages": [
|
||||||
{
|
{
|
||||||
"name": "composer/pcre",
|
"name": "composer/pcre",
|
||||||
"version": "3.1.1",
|
"version": "3.1.3",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/composer/pcre.git",
|
"url": "https://github.com/composer/pcre.git",
|
||||||
"reference": "00104306927c7a0919b4ced2aaa6782c1e61a3c9"
|
"reference": "5b16e25a5355f1f3afdfc2f954a0a80aec4826a8"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/composer/pcre/zipball/00104306927c7a0919b4ced2aaa6782c1e61a3c9",
|
"url": "https://api.github.com/repos/composer/pcre/zipball/5b16e25a5355f1f3afdfc2f954a0a80aec4826a8",
|
||||||
"reference": "00104306927c7a0919b4ced2aaa6782c1e61a3c9",
|
"reference": "5b16e25a5355f1f3afdfc2f954a0a80aec4826a8",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -59,7 +59,7 @@
|
|||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"issues": "https://github.com/composer/pcre/issues",
|
"issues": "https://github.com/composer/pcre/issues",
|
||||||
"source": "https://github.com/composer/pcre/tree/3.1.1"
|
"source": "https://github.com/composer/pcre/tree/3.1.3"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -75,7 +75,7 @@
|
|||||||
"type": "tidelift"
|
"type": "tidelift"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2023-10-11T07:11:09+00:00"
|
"time": "2024-03-19T10:26:25+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "composer/semver",
|
"name": "composer/semver",
|
||||||
@@ -160,16 +160,16 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "composer/xdebug-handler",
|
"name": "composer/xdebug-handler",
|
||||||
"version": "3.0.3",
|
"version": "3.0.4",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/composer/xdebug-handler.git",
|
"url": "https://github.com/composer/xdebug-handler.git",
|
||||||
"reference": "ced299686f41dce890debac69273b47ffe98a40c"
|
"reference": "4f988f8fdf580d53bdb2d1278fe93d1ed5462255"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/composer/xdebug-handler/zipball/ced299686f41dce890debac69273b47ffe98a40c",
|
"url": "https://api.github.com/repos/composer/xdebug-handler/zipball/4f988f8fdf580d53bdb2d1278fe93d1ed5462255",
|
||||||
"reference": "ced299686f41dce890debac69273b47ffe98a40c",
|
"reference": "4f988f8fdf580d53bdb2d1278fe93d1ed5462255",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -180,7 +180,7 @@
|
|||||||
"require-dev": {
|
"require-dev": {
|
||||||
"phpstan/phpstan": "^1.0",
|
"phpstan/phpstan": "^1.0",
|
||||||
"phpstan/phpstan-strict-rules": "^1.1",
|
"phpstan/phpstan-strict-rules": "^1.1",
|
||||||
"symfony/phpunit-bridge": "^6.0"
|
"phpunit/phpunit": "^8.5 || ^9.6 || ^10.5"
|
||||||
},
|
},
|
||||||
"type": "library",
|
"type": "library",
|
||||||
"autoload": {
|
"autoload": {
|
||||||
@@ -204,9 +204,9 @@
|
|||||||
"performance"
|
"performance"
|
||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"irc": "irc://irc.freenode.org/composer",
|
"irc": "ircs://irc.libera.chat:6697/composer",
|
||||||
"issues": "https://github.com/composer/xdebug-handler/issues",
|
"issues": "https://github.com/composer/xdebug-handler/issues",
|
||||||
"source": "https://github.com/composer/xdebug-handler/tree/3.0.3"
|
"source": "https://github.com/composer/xdebug-handler/tree/3.0.4"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -222,20 +222,20 @@
|
|||||||
"type": "tidelift"
|
"type": "tidelift"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2022-02-25T21:32:43+00:00"
|
"time": "2024-03-26T18:29:49+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "friendsofphp/php-cs-fixer",
|
"name": "friendsofphp/php-cs-fixer",
|
||||||
"version": "v3.51.0",
|
"version": "v3.52.1",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer.git",
|
"url": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer.git",
|
||||||
"reference": "127fa74f010da99053e3f5b62672615b72dd6efd"
|
"reference": "6e77207f0d851862ceeb6da63e6e22c01b1587bc"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/PHP-CS-Fixer/PHP-CS-Fixer/zipball/127fa74f010da99053e3f5b62672615b72dd6efd",
|
"url": "https://api.github.com/repos/PHP-CS-Fixer/PHP-CS-Fixer/zipball/6e77207f0d851862ceeb6da63e6e22c01b1587bc",
|
||||||
"reference": "127fa74f010da99053e3f5b62672615b72dd6efd",
|
"reference": "6e77207f0d851862ceeb6da63e6e22c01b1587bc",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -306,7 +306,7 @@
|
|||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"issues": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/issues",
|
"issues": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/issues",
|
||||||
"source": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/tree/v3.51.0"
|
"source": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/tree/v3.52.1"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -314,7 +314,7 @@
|
|||||||
"type": "github"
|
"type": "github"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2024-02-28T19:50:06+00:00"
|
"time": "2024-03-19T21:02:43+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "psr/container",
|
"name": "psr/container",
|
||||||
|
|||||||
12
.ci/phpmd/composer.lock
generated
12
.ci/phpmd/composer.lock
generated
@@ -9,16 +9,16 @@
|
|||||||
"packages-dev": [
|
"packages-dev": [
|
||||||
{
|
{
|
||||||
"name": "composer/pcre",
|
"name": "composer/pcre",
|
||||||
"version": "3.1.1",
|
"version": "3.1.2",
|
||||||
"source": {
|
"source": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/composer/pcre.git",
|
"url": "https://github.com/composer/pcre.git",
|
||||||
"reference": "00104306927c7a0919b4ced2aaa6782c1e61a3c9"
|
"reference": "4775f35b2d70865807c89d32c8e7385b86eb0ace"
|
||||||
},
|
},
|
||||||
"dist": {
|
"dist": {
|
||||||
"type": "zip",
|
"type": "zip",
|
||||||
"url": "https://api.github.com/repos/composer/pcre/zipball/00104306927c7a0919b4ced2aaa6782c1e61a3c9",
|
"url": "https://api.github.com/repos/composer/pcre/zipball/4775f35b2d70865807c89d32c8e7385b86eb0ace",
|
||||||
"reference": "00104306927c7a0919b4ced2aaa6782c1e61a3c9",
|
"reference": "4775f35b2d70865807c89d32c8e7385b86eb0ace",
|
||||||
"shasum": ""
|
"shasum": ""
|
||||||
},
|
},
|
||||||
"require": {
|
"require": {
|
||||||
@@ -60,7 +60,7 @@
|
|||||||
],
|
],
|
||||||
"support": {
|
"support": {
|
||||||
"issues": "https://github.com/composer/pcre/issues",
|
"issues": "https://github.com/composer/pcre/issues",
|
||||||
"source": "https://github.com/composer/pcre/tree/3.1.1"
|
"source": "https://github.com/composer/pcre/tree/3.1.2"
|
||||||
},
|
},
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
@@ -76,7 +76,7 @@
|
|||||||
"type": "tidelift"
|
"type": "tidelift"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"time": "2023-10-11T07:11:09+00:00"
|
"time": "2024-03-07T15:38:35+00:00"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "composer/xdebug-handler",
|
"name": "composer/xdebug-handler",
|
||||||
|
|||||||
10
.env.example
10
.env.example
@@ -111,7 +111,10 @@ PGSQL_SSL_CERT=null
|
|||||||
PGSQL_SSL_KEY=null
|
PGSQL_SSL_KEY=null
|
||||||
PGSQL_SSL_CRL_FILE=null
|
PGSQL_SSL_CRL_FILE=null
|
||||||
|
|
||||||
# more PostgreSQL settings
|
# For postgresql 15 and up, setting this to public will no longer work as expected, becasuse the
|
||||||
|
# 'public' schema is without grants. This can be worked around by having a super user grant those
|
||||||
|
# necessary privileges, but in security conscious setups that's not viable.
|
||||||
|
# You will need to set this to the schema you want to use.
|
||||||
PGSQL_SCHEMA=public
|
PGSQL_SCHEMA=public
|
||||||
|
|
||||||
# If you're looking for performance improvements, you could install memcached or redis
|
# If you're looking for performance improvements, you could install memcached or redis
|
||||||
@@ -184,6 +187,11 @@ SEND_REPORT_JOURNALS=true
|
|||||||
# Since this involves an external service, it's optional and disabled by default.
|
# Since this involves an external service, it's optional and disabled by default.
|
||||||
ENABLE_EXTERNAL_MAP=false
|
ENABLE_EXTERNAL_MAP=false
|
||||||
|
|
||||||
|
#
|
||||||
|
# Enable or disable exchange rate conversion. This function isn't used yet by Firefly III
|
||||||
|
#
|
||||||
|
ENABLE_EXCHANGE_RATES=false
|
||||||
|
|
||||||
# Set this value to true if you want Firefly III to download currency exchange rates
|
# Set this value to true if you want Firefly III to download currency exchange rates
|
||||||
# from the internet. These rates are hosted by the creator of Firefly III inside
|
# from the internet. These rates are hosted by the creator of Firefly III inside
|
||||||
# an Azure Storage Container.
|
# an Azure Storage Container.
|
||||||
|
|||||||
4
.github/funding.yml
vendored
4
.github/funding.yml
vendored
@@ -1,4 +1,6 @@
|
|||||||
# These are supported funding model platforms
|
# Firefly III sponsor options
|
||||||
|
|
||||||
github: jc5
|
github: jc5
|
||||||
patreon: JC5
|
patreon: JC5
|
||||||
|
ko_fi: jamesc5
|
||||||
|
liberapay: JC5
|
||||||
|
|||||||
2
.github/workflows/cleanup.yml
vendored
2
.github/workflows/cleanup.yml
vendored
@@ -7,7 +7,7 @@ permissions:
|
|||||||
|
|
||||||
on:
|
on:
|
||||||
schedule:
|
schedule:
|
||||||
- cron: '0 0 * * *'
|
- cron: '0 1 * * *'
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
jobs:
|
jobs:
|
||||||
prune:
|
prune:
|
||||||
|
|||||||
20
.github/workflows/lock.yml
vendored
20
.github/workflows/lock.yml
vendored
@@ -3,17 +3,27 @@ name: 'Issues - Lock old issues'
|
|||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
schedule:
|
schedule:
|
||||||
- cron: '0 0 * * *'
|
- cron: '0 2 * * *'
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: lock-threads
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
issues: write
|
||||||
|
pull-requests: write
|
||||||
|
discussions: write
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
lock:
|
lock:
|
||||||
permissions:
|
permissions:
|
||||||
issues: write
|
issues: write
|
||||||
pull-requests: write
|
pull-requests: write
|
||||||
|
discussions: write
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: JC5/lock-threads@main
|
- uses: dessant/lock-threads@v5
|
||||||
with:
|
with:
|
||||||
github-token: ${{ github.token }}
|
issue-inactive-days: 21
|
||||||
issue-inactive-days: 7
|
pr-inactive-days: 21
|
||||||
pr-inactive-days: 7
|
discussion-inactive-days: 21
|
||||||
|
log-output: true
|
||||||
|
|||||||
100
.github/workflows/release.yml
vendored
100
.github/workflows/release.yml
vendored
@@ -8,7 +8,7 @@ on:
|
|||||||
required: true
|
required: true
|
||||||
default: 'develop'
|
default: 'develop'
|
||||||
schedule:
|
schedule:
|
||||||
- cron: '15 0 * * MON,THU'
|
- cron: '0 3 * * MON,THU'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
@@ -51,7 +51,7 @@ jobs:
|
|||||||
CROWDIN_TOKEN: ${{ secrets.CROWDIN_TOKEN }}
|
CROWDIN_TOKEN: ${{ secrets.CROWDIN_TOKEN }}
|
||||||
- name: Cleanup translations
|
- name: Cleanup translations
|
||||||
id: cleanup-transactions
|
id: cleanup-transactions
|
||||||
uses: JC5/firefly-iii-dev@v34
|
uses: JC5/firefly-iii-dev@v36
|
||||||
with:
|
with:
|
||||||
action: 'ff3:crowdin-warning'
|
action: 'ff3:crowdin-warning'
|
||||||
output: ''
|
output: ''
|
||||||
@@ -60,7 +60,7 @@ jobs:
|
|||||||
GH_TOKEN: ''
|
GH_TOKEN: ''
|
||||||
- name: Cleanup changelog
|
- name: Cleanup changelog
|
||||||
id: cleanup-changelog
|
id: cleanup-changelog
|
||||||
uses: JC5/firefly-iii-dev@v34
|
uses: JC5/firefly-iii-dev@v36
|
||||||
with:
|
with:
|
||||||
action: 'ff3:changelog'
|
action: 'ff3:changelog'
|
||||||
output: ''
|
output: ''
|
||||||
@@ -69,7 +69,7 @@ jobs:
|
|||||||
GH_TOKEN: ${{ secrets.CHANGELOG_TOKEN }}
|
GH_TOKEN: ${{ secrets.CHANGELOG_TOKEN }}
|
||||||
- name: Extract changelog
|
- name: Extract changelog
|
||||||
id: extract-changelog
|
id: extract-changelog
|
||||||
uses: JC5/firefly-iii-dev@v34
|
uses: JC5/firefly-iii-dev@v36
|
||||||
with:
|
with:
|
||||||
action: 'ff3:extract-changelog'
|
action: 'ff3:extract-changelog'
|
||||||
output: 'output'
|
output: 'output'
|
||||||
@@ -78,7 +78,7 @@ jobs:
|
|||||||
GH_TOKEN: ""
|
GH_TOKEN: ""
|
||||||
- name: Replace version
|
- name: Replace version
|
||||||
id: replace-version
|
id: replace-version
|
||||||
uses: JC5/firefly-iii-dev@v34
|
uses: JC5/firefly-iii-dev@v36
|
||||||
with:
|
with:
|
||||||
action: 'ff3:version'
|
action: 'ff3:version'
|
||||||
output: ''
|
output: ''
|
||||||
@@ -88,7 +88,7 @@ jobs:
|
|||||||
FF_III_VERSION: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }}
|
FF_III_VERSION: ${{ github.event_name == 'schedule' && 'develop' || github.event.inputs.version }}
|
||||||
- name: Generate JSON v1
|
- name: Generate JSON v1
|
||||||
id: json-v1
|
id: json-v1
|
||||||
uses: JC5/firefly-iii-dev@v34
|
uses: JC5/firefly-iii-dev@v36
|
||||||
with:
|
with:
|
||||||
action: 'ff3:json-translations v1'
|
action: 'ff3:json-translations v1'
|
||||||
output: ''
|
output: ''
|
||||||
@@ -97,7 +97,7 @@ jobs:
|
|||||||
GH_TOKEN: ''
|
GH_TOKEN: ''
|
||||||
- name: Generate JSON v2
|
- name: Generate JSON v2
|
||||||
id: json-v2
|
id: json-v2
|
||||||
uses: JC5/firefly-iii-dev@v34
|
uses: JC5/firefly-iii-dev@v36
|
||||||
with:
|
with:
|
||||||
action: 'ff3:json-translations v2'
|
action: 'ff3:json-translations v2'
|
||||||
output: ''
|
output: ''
|
||||||
@@ -106,7 +106,7 @@ jobs:
|
|||||||
GH_TOKEN: ''
|
GH_TOKEN: ''
|
||||||
- name: Code cleanup
|
- name: Code cleanup
|
||||||
id: code-cleanup
|
id: code-cleanup
|
||||||
uses: JC5/firefly-iii-dev@v34
|
uses: JC5/firefly-iii-dev@v36
|
||||||
with:
|
with:
|
||||||
action: 'ff3:code'
|
action: 'ff3:code'
|
||||||
output: ''
|
output: ''
|
||||||
@@ -115,11 +115,12 @@ jobs:
|
|||||||
GH_TOKEN: ''
|
GH_TOKEN: ''
|
||||||
- name: Build new JS
|
- name: Build new JS
|
||||||
run: |
|
run: |
|
||||||
npm upgrade
|
npm install
|
||||||
|
npm update
|
||||||
npm run build
|
npm run build
|
||||||
- name: Build old JS
|
- name: Build old JS
|
||||||
id: old-js
|
id: old-js
|
||||||
uses: JC5/firefly-iii-dev@v34
|
uses: JC5/firefly-iii-dev@v36
|
||||||
with:
|
with:
|
||||||
action: 'ff3:old-js'
|
action: 'ff3:old-js'
|
||||||
output: ''
|
output: ''
|
||||||
@@ -140,8 +141,20 @@ jobs:
|
|||||||
git config user.email 41898282+github-actions[bot]@users.noreply.github.com
|
git config user.email 41898282+github-actions[bot]@users.noreply.github.com
|
||||||
git config advice.addIgnoredFile false
|
git config advice.addIgnoredFile false
|
||||||
|
|
||||||
|
# update composer (again)
|
||||||
|
composer validate --strict
|
||||||
|
composer update --no-dev --no-scripts --no-plugins
|
||||||
|
composer dump-autoload
|
||||||
|
|
||||||
|
releaseName=$version
|
||||||
|
zipName=FireflyIII-$version.zip
|
||||||
|
tarName=FireflyIII-$version.tar.gz
|
||||||
|
|
||||||
if [[ "develop" == "$version" ]]; then
|
if [[ "develop" == "$version" ]]; then
|
||||||
[[ -z $(git status --untracked-files=normal --porcelain) ]] && echo "this branch is clean, no need to push..." && exit 0;
|
[[ -z $(git status --untracked-files=normal --porcelain) ]] && echo "this branch is clean, no need to push..." && exit 0;
|
||||||
|
releaseName=$version-$(date +'%Y%m%d')
|
||||||
|
zipName=FireflyIII-develop.zip
|
||||||
|
tarName=FireflyIII-develop.tar.gz
|
||||||
fi
|
fi
|
||||||
|
|
||||||
git add -A
|
git add -A
|
||||||
@@ -151,19 +164,64 @@ jobs:
|
|||||||
git commit -m "Auto commit for release '$version' on $(date +'%Y-%m-%d')" || true
|
git commit -m "Auto commit for release '$version' on $(date +'%Y-%m-%d')" || true
|
||||||
git push
|
git push
|
||||||
|
|
||||||
|
# zip and tar everything
|
||||||
|
zip -rq $zipName . -x "*.git*" "*.ci*" "*.github*" "*node_modules*" "*output.txt*"
|
||||||
|
touch $tarName
|
||||||
|
tar --exclude=$tarName --exclude='./.git' --exclude='./.ci' --exclude='./.github' --exclude='./node_modules' --exclude='./output.txt' -czf $tarName .
|
||||||
|
|
||||||
|
# add sha256 sum
|
||||||
|
sha256sum -b $zipName > $zipName.sha256
|
||||||
|
sha256sum -b $tarName > $tarName.sha256
|
||||||
|
|
||||||
if [[ "develop" == "$version" ]]; then
|
if [[ "develop" == "$version" ]]; then
|
||||||
echo "Create nightly release."
|
# add text to output.txt (instructions)
|
||||||
git tag -a $version-$(date +'%Y%m%d') -m "Nightly development release '$version' on $(date +'%Y-%m-%d')"
|
|
||||||
git push origin $version-$(date +'%Y%m%d')
|
|
||||||
gh release create $version-$(date +'%Y%m%d') -p --verify-tag \
|
|
||||||
-t "Development release for $(date +'%Y-%m-%d')" \
|
|
||||||
-n "Bi-weekly development release of Firefly III with the latest fixes, translations and features. This release was created on **$(date +'%Y-%m-%d')** and may contain bugs. Use at your own risk. Docker users can find this release under the \`develop\` tag."
|
|
||||||
else
|
|
||||||
echo "Create default release."
|
|
||||||
git tag -a $version -m "Here be changelog"
|
|
||||||
git push origin $version
|
|
||||||
gh release create $version -F output.txt -t "$version" --verify-tag
|
|
||||||
rm output.txt
|
rm output.txt
|
||||||
|
echo "Bi-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')** and may contain unexpected bugs. Data loss is rare but is not impossible." >> 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 is may not work as expected." >> output.txt
|
||||||
|
|
||||||
|
# create the release:
|
||||||
|
echo "Create nightly release."
|
||||||
|
git tag -a $releaseName -m "Nightly development release '$version' on $(date +'%Y-%m-%d')"
|
||||||
|
git push origin $releaseName
|
||||||
|
gh release create $releaseName -p --verify-tag \
|
||||||
|
-t "Development release for $(date +'%Y-%m-%d')" \
|
||||||
|
-F output.txt
|
||||||
|
|
||||||
|
# add zip file to release.
|
||||||
|
gh release upload $releaseName $zipName
|
||||||
|
gh release upload $releaseName $tarName
|
||||||
|
|
||||||
|
# add sha256 sum to release
|
||||||
|
gh release upload $releaseName $zipName.sha256
|
||||||
|
gh release upload $releaseName $tarName.sha256
|
||||||
|
|
||||||
|
# rm output.txt again
|
||||||
|
rm output.txt
|
||||||
|
else
|
||||||
|
# add text to output.txt (more instructions)
|
||||||
|
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 "Create default release."
|
||||||
|
git tag -a $releaseName -m "Here be changelog"
|
||||||
|
git push origin $releaseName
|
||||||
|
gh release create $releaseName -F output.txt -t "$releaseName" --verify-tag
|
||||||
|
# add zip file to release.
|
||||||
|
gh release upload $releaseName $zipName
|
||||||
|
# add sha256 sum to release
|
||||||
|
gh release upload $releaseName $zipName.sha256
|
||||||
|
rm output.txt
|
||||||
|
rm $zipName
|
||||||
|
rm $zipName.sha256
|
||||||
git checkout develop
|
git checkout develop
|
||||||
git merge main
|
git merge main
|
||||||
git push
|
git push
|
||||||
|
|||||||
9
.github/workflows/sonarcloud.yml
vendored
9
.github/workflows/sonarcloud.yml
vendored
@@ -45,15 +45,6 @@ jobs:
|
|||||||
- name: Install Composer dependencies
|
- name: Install Composer dependencies
|
||||||
run: composer install --prefer-dist --no-interaction --no-progress --no-scripts
|
run: composer install --prefer-dist --no-interaction --no-progress --no-scripts
|
||||||
|
|
||||||
- name: PHPStan
|
|
||||||
run: .ci/phpstan.sh
|
|
||||||
|
|
||||||
- name: PHPMD
|
|
||||||
run: .ci/phpmd.sh
|
|
||||||
|
|
||||||
- name: PHP CS Fixer
|
|
||||||
run: .ci/phpcs.sh
|
|
||||||
|
|
||||||
- name: "Create database file"
|
- name: "Create database file"
|
||||||
run: touch storage/database/database.sqlite
|
run: touch storage/database/database.sqlite
|
||||||
|
|
||||||
|
|||||||
2
.github/workflows/stale.yml
vendored
2
.github/workflows/stale.yml
vendored
@@ -1,7 +1,7 @@
|
|||||||
name: "Issues - Mark and close stale issues"
|
name: "Issues - Mark and close stale issues"
|
||||||
on:
|
on:
|
||||||
schedule:
|
schedule:
|
||||||
- cron: "30 1 * * *"
|
- cron: "0 4 * * *"
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
|
|||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -7,3 +7,6 @@ yarn-error.log
|
|||||||
.env
|
.env
|
||||||
/.ci/php-cs-fixer/vendor
|
/.ci/php-cs-fixer/vendor
|
||||||
coverage.xml
|
coverage.xml
|
||||||
|
|
||||||
|
# ignore generated files.
|
||||||
|
public/build
|
||||||
|
|||||||
47
app/Api/V1/Controllers/Models/Rule/ExpressionController.php
Normal file
47
app/Api/V1/Controllers/Models/Rule/ExpressionController.php
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
<?php
|
||||||
|
/*
|
||||||
|
* ExpressionController.php
|
||||||
|
* Copyright (c) 2024 Michael Thomas
|
||||||
|
*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace FireflyIII\Api\V1\Controllers\Models\Rule;
|
||||||
|
|
||||||
|
use FireflyIII\Api\V1\Controllers\Controller;
|
||||||
|
use FireflyIII\Api\V1\Requests\Models\Rule\ValidateExpressionRequest;
|
||||||
|
use Illuminate\Http\JsonResponse;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class ExpressionController
|
||||||
|
*/
|
||||||
|
class ExpressionController extends Controller
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* This endpoint is documented at:
|
||||||
|
* https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/rules/validateExpression
|
||||||
|
*
|
||||||
|
* @SuppressWarnings(PHPMD.UnusedFormalParameter)
|
||||||
|
*/
|
||||||
|
public function validateExpression(ValidateExpressionRequest $request): JsonResponse
|
||||||
|
{
|
||||||
|
return response()->json([
|
||||||
|
'valid' => true,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -281,7 +281,7 @@ class BasicController extends Controller
|
|||||||
$spentInCurrency = $row['sum'];
|
$spentInCurrency = $row['sum'];
|
||||||
$leftToSpend = bcadd($amount, $spentInCurrency);
|
$leftToSpend = bcadd($amount, $spentInCurrency);
|
||||||
|
|
||||||
$days = $today->diffInDays($end) + 1;
|
$days = (int)$today->diffInDays($end, true) + 1;
|
||||||
$perDay = '0';
|
$perDay = '0';
|
||||||
if (0 !== $days && bccomp($leftToSpend, '0') > -1) {
|
if (0 !== $days && bccomp($leftToSpend, '0') > -1) {
|
||||||
$perDay = bcdiv($leftToSpend, (string)$days);
|
$perDay = bcdiv($leftToSpend, (string)$days);
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ class DateRequest extends FormRequest
|
|||||||
{
|
{
|
||||||
$start = $this->getCarbonDate('start');
|
$start = $this->getCarbonDate('start');
|
||||||
$end = $this->getCarbonDate('end');
|
$end = $this->getCarbonDate('end');
|
||||||
if ($start->diffInYears($end) > 5) {
|
if ($start->diffInYears($end, true) > 5) {
|
||||||
throw new FireflyException('Date range out of range.');
|
throw new FireflyException('Date range out of range.');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ declare(strict_types=1);
|
|||||||
namespace FireflyIII\Api\V1\Requests\Models\Rule;
|
namespace FireflyIII\Api\V1\Requests\Models\Rule;
|
||||||
|
|
||||||
use FireflyIII\Rules\IsBoolean;
|
use FireflyIII\Rules\IsBoolean;
|
||||||
|
use FireflyIII\Rules\IsValidActionExpression;
|
||||||
use FireflyIII\Support\Request\ChecksLogin;
|
use FireflyIII\Support\Request\ChecksLogin;
|
||||||
use FireflyIII\Support\Request\ConvertsDataTypes;
|
use FireflyIII\Support\Request\ConvertsDataTypes;
|
||||||
use FireflyIII\Support\Request\GetRuleConfiguration;
|
use FireflyIII\Support\Request\GetRuleConfiguration;
|
||||||
@@ -57,7 +58,6 @@ class StoreRequest extends FormRequest
|
|||||||
'active' => ['active', 'boolean'],
|
'active' => ['active', 'boolean'],
|
||||||
];
|
];
|
||||||
$data = $this->getAllData($fields);
|
$data = $this->getAllData($fields);
|
||||||
|
|
||||||
$data['triggers'] = $this->getRuleTriggers();
|
$data['triggers'] = $this->getRuleTriggers();
|
||||||
$data['actions'] = $this->getRuleActions();
|
$data['actions'] = $this->getRuleActions();
|
||||||
|
|
||||||
@@ -123,7 +123,7 @@ class StoreRequest extends FormRequest
|
|||||||
'triggers.*.stop_processing' => [new IsBoolean()],
|
'triggers.*.stop_processing' => [new IsBoolean()],
|
||||||
'triggers.*.active' => [new IsBoolean()],
|
'triggers.*.active' => [new IsBoolean()],
|
||||||
'actions.*.type' => 'required|in:'.implode(',', $validActions),
|
'actions.*.type' => 'required|in:'.implode(',', $validActions),
|
||||||
'actions.*.value' => 'required_if:actions.*.type,'.$contextActions.'|ruleActionValue',
|
'actions.*.value' => [sprintf('required_if:actions.*.type,%s', $contextActions), new IsValidActionExpression(), 'ruleActionValue'],
|
||||||
'actions.*.stop_processing' => [new IsBoolean()],
|
'actions.*.stop_processing' => [new IsBoolean()],
|
||||||
'actions.*.active' => [new IsBoolean()],
|
'actions.*.active' => [new IsBoolean()],
|
||||||
'strict' => [new IsBoolean()],
|
'strict' => [new IsBoolean()],
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ namespace FireflyIII\Api\V1\Requests\Models\Rule;
|
|||||||
|
|
||||||
use FireflyIII\Models\Rule;
|
use FireflyIII\Models\Rule;
|
||||||
use FireflyIII\Rules\IsBoolean;
|
use FireflyIII\Rules\IsBoolean;
|
||||||
|
use FireflyIII\Rules\IsValidActionExpression;
|
||||||
use FireflyIII\Support\Request\ChecksLogin;
|
use FireflyIII\Support\Request\ChecksLogin;
|
||||||
use FireflyIII\Support\Request\ConvertsDataTypes;
|
use FireflyIII\Support\Request\ConvertsDataTypes;
|
||||||
use FireflyIII\Support\Request\GetRuleConfiguration;
|
use FireflyIII\Support\Request\GetRuleConfiguration;
|
||||||
@@ -140,7 +141,7 @@ class UpdateRequest extends FormRequest
|
|||||||
'triggers.*.stop_processing' => [new IsBoolean()],
|
'triggers.*.stop_processing' => [new IsBoolean()],
|
||||||
'triggers.*.active' => [new IsBoolean()],
|
'triggers.*.active' => [new IsBoolean()],
|
||||||
'actions.*.type' => 'required|in:'.implode(',', $validActions),
|
'actions.*.type' => 'required|in:'.implode(',', $validActions),
|
||||||
'actions.*.value' => 'required_if:actions.*.type,'.$contextActions.'|ruleActionValue',
|
'actions.*.value' => [sprintf('required_if:actions.*.type,%s', $contextActions), new IsValidActionExpression(), 'ruleActionValue'],
|
||||||
'actions.*.stop_processing' => [new IsBoolean()],
|
'actions.*.stop_processing' => [new IsBoolean()],
|
||||||
'actions.*.active' => [new IsBoolean()],
|
'actions.*.active' => [new IsBoolean()],
|
||||||
'strict' => [new IsBoolean()],
|
'strict' => [new IsBoolean()],
|
||||||
|
|||||||
@@ -0,0 +1,42 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ValidateExpressionRequest.php
|
||||||
|
* Copyright (c) 2024 Michael Thomas
|
||||||
|
*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace FireflyIII\Api\V1\Requests\Models\Rule;
|
||||||
|
|
||||||
|
use FireflyIII\Rules\IsValidActionExpression;
|
||||||
|
use FireflyIII\Support\Request\ChecksLogin;
|
||||||
|
use Illuminate\Foundation\Http\FormRequest;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Class ValidateExpressionRequest
|
||||||
|
*/
|
||||||
|
class ValidateExpressionRequest extends FormRequest
|
||||||
|
{
|
||||||
|
use ChecksLogin;
|
||||||
|
|
||||||
|
public function rules(): array
|
||||||
|
{
|
||||||
|
return ['expression' => ['required', new IsValidActionExpression()]];
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -158,7 +158,8 @@ class Controller extends BaseController
|
|||||||
|
|
||||||
// the transformer, at this point, needs to collect information that ALL items in the collection
|
// the transformer, at this point, needs to collect information that ALL items in the collection
|
||||||
// require, like meta-data and stuff like that, and save it for later.
|
// require, like meta-data and stuff like that, and save it for later.
|
||||||
$transformer->collectMetaData($objects);
|
$objects = $transformer->collectMetaData($objects);
|
||||||
|
$paginator->setCollection($objects);
|
||||||
|
|
||||||
$resource = new FractalCollection($objects, $transformer, $key);
|
$resource = new FractalCollection($objects, $transformer, $key);
|
||||||
$resource->setPaginator(new IlluminatePaginatorAdapter($paginator));
|
$resource->setPaginator(new IlluminatePaginatorAdapter($paginator));
|
||||||
|
|||||||
@@ -64,12 +64,15 @@ class IndexController extends Controller
|
|||||||
{
|
{
|
||||||
$this->repository->resetAccountOrder();
|
$this->repository->resetAccountOrder();
|
||||||
$types = $request->getAccountTypes();
|
$types = $request->getAccountTypes();
|
||||||
$accounts = $this->repository->getAccountsByType($types);
|
$instructions = $request->getSortInstructions('accounts');
|
||||||
|
$accounts = $this->repository->getAccountsByType($types, $instructions);
|
||||||
$pageSize = $this->parameters->get('limit');
|
$pageSize = $this->parameters->get('limit');
|
||||||
$count = $accounts->count();
|
$count = $accounts->count();
|
||||||
$accounts = $accounts->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize);
|
$accounts = $accounts->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize);
|
||||||
$paginator = new LengthAwarePaginator($accounts, $count, $pageSize, $this->parameters->get('page'));
|
$paginator = new LengthAwarePaginator($accounts, $count, $pageSize, $this->parameters->get('page'));
|
||||||
$transformer = new AccountTransformer();
|
$transformer = new AccountTransformer();
|
||||||
|
|
||||||
|
$this->parameters->set('sort', $instructions);
|
||||||
$transformer->setParameters($this->parameters); // give params to transformer
|
$transformer->setParameters($this->parameters); // give params to transformer
|
||||||
|
|
||||||
return response()
|
return response()
|
||||||
|
|||||||
79
app/Api/V2/Controllers/Model/Account/UpdateController.php
Normal file
79
app/Api/V2/Controllers/Model/Account/UpdateController.php
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
<?php
|
||||||
|
/*
|
||||||
|
* UpdateController.php
|
||||||
|
* Copyright (c) 2024 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace FireflyIII\Api\V2\Controllers\Model\Account;
|
||||||
|
|
||||||
|
use FireflyIII\Api\V2\Controllers\Controller;
|
||||||
|
use FireflyIII\Api\V2\Request\Model\Account\UpdateRequest;
|
||||||
|
use FireflyIII\Models\Account;
|
||||||
|
use FireflyIII\Repositories\UserGroups\Account\AccountRepositoryInterface;
|
||||||
|
use FireflyIII\Transformers\V2\AccountTransformer;
|
||||||
|
use Illuminate\Http\JsonResponse;
|
||||||
|
|
||||||
|
class UpdateController extends Controller
|
||||||
|
{
|
||||||
|
public const string RESOURCE_KEY = 'accounts';
|
||||||
|
|
||||||
|
private AccountRepositoryInterface $repository;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* AccountController constructor.
|
||||||
|
*/
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
parent::__construct();
|
||||||
|
$this->middleware(
|
||||||
|
function ($request, $next) {
|
||||||
|
$this->repository = app(AccountRepositoryInterface::class);
|
||||||
|
// new way of user group validation
|
||||||
|
$userGroup = $this->validateUserGroup($request);
|
||||||
|
if (null !== $userGroup) {
|
||||||
|
$this->repository->setUserGroup($userGroup);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $next($request);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TODO this endpoint is not yet reachable.
|
||||||
|
*/
|
||||||
|
public function update(UpdateRequest $request, Account $account): JsonResponse
|
||||||
|
{
|
||||||
|
app('log')->debug(sprintf('Now in %s', __METHOD__));
|
||||||
|
$data = $request->getUpdateData();
|
||||||
|
$data['type'] = config('firefly.shortNamesByFullName.'.$account->accountType->type);
|
||||||
|
$account = $this->repository->update($account, $data);
|
||||||
|
$account->refresh();
|
||||||
|
app('preferences')->mark();
|
||||||
|
|
||||||
|
$transformer = new AccountTransformer();
|
||||||
|
$transformer->setParameters($this->parameters);
|
||||||
|
|
||||||
|
return response()
|
||||||
|
->api($this->jsonApiObject('accounts', $account, $transformer))
|
||||||
|
->header('Content-Type', self::CONTENT_TYPE)
|
||||||
|
;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -298,7 +298,7 @@ class BasicController extends Controller
|
|||||||
app('log')->debug(sprintf('Amount left is %s', $left));
|
app('log')->debug(sprintf('Amount left is %s', $left));
|
||||||
|
|
||||||
// how much left per day?
|
// how much left per day?
|
||||||
$days = $today->diffInDays($end) + 1;
|
$days = (int) $today->diffInDays($end, true) + 1;
|
||||||
$perDay = '0';
|
$perDay = '0';
|
||||||
$perDayNative = '0';
|
$perDayNative = '0';
|
||||||
if (0 !== $days && bccomp($left, '0') > -1) {
|
if (0 !== $days && bccomp($left, '0') > -1) {
|
||||||
|
|||||||
@@ -27,6 +27,7 @@ use Carbon\Carbon;
|
|||||||
use FireflyIII\Support\Http\Api\AccountFilter;
|
use FireflyIII\Support\Http\Api\AccountFilter;
|
||||||
use FireflyIII\Support\Request\ChecksLogin;
|
use FireflyIII\Support\Request\ChecksLogin;
|
||||||
use FireflyIII\Support\Request\ConvertsDataTypes;
|
use FireflyIII\Support\Request\ConvertsDataTypes;
|
||||||
|
use FireflyIII\Support\Request\GetSortInstructions;
|
||||||
use Illuminate\Foundation\Http\FormRequest;
|
use Illuminate\Foundation\Http\FormRequest;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -39,6 +40,14 @@ class IndexRequest extends FormRequest
|
|||||||
use AccountFilter;
|
use AccountFilter;
|
||||||
use ChecksLogin;
|
use ChecksLogin;
|
||||||
use ConvertsDataTypes;
|
use ConvertsDataTypes;
|
||||||
|
use GetSortInstructions;
|
||||||
|
|
||||||
|
public function getAccountTypes(): array
|
||||||
|
{
|
||||||
|
$type = (string)$this->get('type', 'default');
|
||||||
|
|
||||||
|
return $this->mapAccountTypes($type);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get all data from the request.
|
* Get all data from the request.
|
||||||
@@ -48,13 +57,6 @@ class IndexRequest extends FormRequest
|
|||||||
return $this->getCarbonDate('date');
|
return $this->getCarbonDate('date');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getAccountTypes(): array
|
|
||||||
{
|
|
||||||
$type = (string)$this->get('type', 'default');
|
|
||||||
|
|
||||||
return $this->mapAccountTypes($type);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The rules that the incoming request must be matched against.
|
* The rules that the incoming request must be matched against.
|
||||||
*/
|
*/
|
||||||
|
|||||||
116
app/Api/V2/Request/Model/Account/UpdateRequest.php
Normal file
116
app/Api/V2/Request/Model/Account/UpdateRequest.php
Normal file
@@ -0,0 +1,116 @@
|
|||||||
|
<?php
|
||||||
|
/*
|
||||||
|
* UpdateRequest.php
|
||||||
|
* Copyright (c) 2024 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace FireflyIII\Api\V2\Request\Model\Account;
|
||||||
|
|
||||||
|
use FireflyIII\Models\Account;
|
||||||
|
use FireflyIII\Models\Location;
|
||||||
|
use FireflyIII\Rules\IsBoolean;
|
||||||
|
use FireflyIII\Rules\UniqueAccountNumber;
|
||||||
|
use FireflyIII\Rules\UniqueIban;
|
||||||
|
use FireflyIII\Support\Request\AppendsLocationData;
|
||||||
|
use FireflyIII\Support\Request\ChecksLogin;
|
||||||
|
use FireflyIII\Support\Request\ConvertsDataTypes;
|
||||||
|
use Illuminate\Foundation\Http\FormRequest;
|
||||||
|
|
||||||
|
class UpdateRequest extends FormRequest
|
||||||
|
{
|
||||||
|
use AppendsLocationData;
|
||||||
|
use ChecksLogin;
|
||||||
|
use ConvertsDataTypes;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TODO is a duplicate of the v1 update thing.
|
||||||
|
*/
|
||||||
|
public function getUpdateData(): array
|
||||||
|
{
|
||||||
|
$fields = [
|
||||||
|
'name' => ['name', 'convertString'],
|
||||||
|
'active' => ['active', 'boolean'],
|
||||||
|
'include_net_worth' => ['include_net_worth', 'boolean'],
|
||||||
|
'account_type_name' => ['type', 'convertString'],
|
||||||
|
'virtual_balance' => ['virtual_balance', 'convertString'],
|
||||||
|
'iban' => ['iban', 'convertString'],
|
||||||
|
'BIC' => ['bic', 'convertString'],
|
||||||
|
'account_number' => ['account_number', 'convertString'],
|
||||||
|
'account_role' => ['account_role', 'convertString'],
|
||||||
|
'liability_type' => ['liability_type', 'convertString'],
|
||||||
|
'opening_balance' => ['opening_balance', 'convertString'],
|
||||||
|
'opening_balance_date' => ['opening_balance_date', 'convertDateTime'],
|
||||||
|
'cc_type' => ['credit_card_type', 'convertString'],
|
||||||
|
'cc_monthly_payment_date' => ['monthly_payment_date', 'convertDateTime'],
|
||||||
|
'notes' => ['notes', 'stringWithNewlines'],
|
||||||
|
'interest' => ['interest', 'convertString'],
|
||||||
|
'interest_period' => ['interest_period', 'convertString'],
|
||||||
|
'order' => ['order', 'convertInteger'],
|
||||||
|
'currency_id' => ['currency_id', 'convertInteger'],
|
||||||
|
'currency_code' => ['currency_code', 'convertString'],
|
||||||
|
'liability_direction' => ['liability_direction', 'convertString'],
|
||||||
|
'liability_amount' => ['liability_amount', 'convertString'],
|
||||||
|
'liability_start_date' => ['liability_start_date', 'date'],
|
||||||
|
];
|
||||||
|
$data = $this->getAllData($fields);
|
||||||
|
|
||||||
|
return $this->appendLocationData($data, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TODO is a duplicate of the v1 UpdateRequest method.
|
||||||
|
*
|
||||||
|
* The rules that the incoming request must be matched against.
|
||||||
|
*/
|
||||||
|
public function rules(): array
|
||||||
|
{
|
||||||
|
/** @var Account $account */
|
||||||
|
$account = $this->route()->parameter('account');
|
||||||
|
$accountRoles = implode(',', config('firefly.accountRoles'));
|
||||||
|
$types = implode(',', array_keys(config('firefly.subTitlesByIdentifier')));
|
||||||
|
$ccPaymentTypes = implode(',', array_keys(config('firefly.ccTypes')));
|
||||||
|
|
||||||
|
$rules = [
|
||||||
|
'name' => sprintf('min:1|max:1024|uniqueAccountForUser:%d', $account->id),
|
||||||
|
'type' => sprintf('in:%s', $types),
|
||||||
|
'iban' => ['iban', 'nullable', new UniqueIban($account, $this->convertString('type'))],
|
||||||
|
'bic' => 'bic|nullable',
|
||||||
|
'account_number' => ['min:1', 'max:255', 'nullable', new UniqueAccountNumber($account, $this->convertString('type'))],
|
||||||
|
'opening_balance' => 'numeric|required_with:opening_balance_date|nullable',
|
||||||
|
'opening_balance_date' => 'date|required_with:opening_balance|nullable',
|
||||||
|
'virtual_balance' => 'numeric|nullable',
|
||||||
|
'order' => 'numeric|nullable',
|
||||||
|
'currency_id' => 'numeric|exists:transaction_currencies,id',
|
||||||
|
'currency_code' => 'min:3|max:51|exists:transaction_currencies,code',
|
||||||
|
'active' => [new IsBoolean()],
|
||||||
|
'include_net_worth' => [new IsBoolean()],
|
||||||
|
'account_role' => sprintf('in:%s|nullable|required_if:type,asset', $accountRoles),
|
||||||
|
'credit_card_type' => sprintf('in:%s|nullable|required_if:account_role,ccAsset', $ccPaymentTypes),
|
||||||
|
'monthly_payment_date' => 'date|nullable|required_if:account_role,ccAsset|required_if:credit_card_type,monthlyFull',
|
||||||
|
'liability_type' => 'required_if:type,liability|in:loan,debt,mortgage',
|
||||||
|
'liability_direction' => 'required_if:type,liability|in:credit,debit',
|
||||||
|
'interest' => 'required_if:type,liability|min:0|max:100|numeric',
|
||||||
|
'interest_period' => 'required_if:type,liability|in:daily,monthly,yearly',
|
||||||
|
'notes' => 'min:0|max:32768',
|
||||||
|
];
|
||||||
|
|
||||||
|
return Location::requestRules($rules);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -29,6 +29,7 @@ use FireflyIII\Support\Http\Api\AccountFilter;
|
|||||||
use FireflyIII\Support\Http\Api\TransactionFilter;
|
use FireflyIII\Support\Http\Api\TransactionFilter;
|
||||||
use FireflyIII\Support\Request\ChecksLogin;
|
use FireflyIII\Support\Request\ChecksLogin;
|
||||||
use FireflyIII\Support\Request\ConvertsDataTypes;
|
use FireflyIII\Support\Request\ConvertsDataTypes;
|
||||||
|
use FireflyIII\Support\Request\GetSortInstructions;
|
||||||
use Illuminate\Foundation\Http\FormRequest;
|
use Illuminate\Foundation\Http\FormRequest;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -40,6 +41,7 @@ class InfiniteListRequest extends FormRequest
|
|||||||
use AccountFilter;
|
use AccountFilter;
|
||||||
use ChecksLogin;
|
use ChecksLogin;
|
||||||
use ConvertsDataTypes;
|
use ConvertsDataTypes;
|
||||||
|
use GetSortInstructions;
|
||||||
use TransactionFilter;
|
use TransactionFilter;
|
||||||
|
|
||||||
public function buildParams(): string
|
public function buildParams(): string
|
||||||
@@ -97,31 +99,6 @@ class InfiniteListRequest extends FormRequest
|
|||||||
return 0 === $page || $page > 65536 ? 1 : $page;
|
return 0 === $page || $page > 65536 ? 1 : $page;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getSortInstructions(string $key): array
|
|
||||||
{
|
|
||||||
$allowed = config(sprintf('firefly.sorting.allowed.%s', $key));
|
|
||||||
$set = $this->get('sorting', []);
|
|
||||||
$result = [];
|
|
||||||
if (0 === count($set)) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
foreach ($set as $info) {
|
|
||||||
$column = $info['column'] ?? 'NOPE';
|
|
||||||
$direction = $info['direction'] ?? 'NOPE';
|
|
||||||
if ('asc' !== $direction && 'desc' !== $direction) {
|
|
||||||
// skip invalid direction
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (false === in_array($column, $allowed, true)) {
|
|
||||||
// skip invalid column
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
$result[$column] = $direction;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $result;
|
|
||||||
}
|
|
||||||
|
|
||||||
public function getTransactionTypes(): array
|
public function getTransactionTypes(): array
|
||||||
{
|
{
|
||||||
$type = (string)$this->get('type', 'default');
|
$type = (string)$this->get('type', 'default');
|
||||||
|
|||||||
65
app/Console/Commands/System/LaravelPassportKeys.php
Normal file
65
app/Console/Commands/System/LaravelPassportKeys.php
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
/*
|
||||||
|
* LaravelPassportKeys.php
|
||||||
|
* Copyright (c) 2024 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\Console\Commands\System;
|
||||||
|
|
||||||
|
use FireflyIII\Console\Commands\ShowsFriendlyMessages;
|
||||||
|
use Illuminate\Console\Command;
|
||||||
|
use Illuminate\Support\Facades\Artisan;
|
||||||
|
use Symfony\Component\Console\Command\Command as CommandAlias;
|
||||||
|
|
||||||
|
class LaravelPassportKeys extends Command
|
||||||
|
{
|
||||||
|
use ShowsFriendlyMessages;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The name and signature of the console command.
|
||||||
|
*
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
protected $signature = 'firefly-iii:laravel-passport-keys';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The console command description.
|
||||||
|
*
|
||||||
|
* @var string
|
||||||
|
*/
|
||||||
|
protected $description = 'Calls the Laravel "passport:keys" but doesn\'t exit 1.';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the console command.
|
||||||
|
*/
|
||||||
|
public function handle()
|
||||||
|
{
|
||||||
|
Artisan::call('passport:keys --no-interaction', []);
|
||||||
|
$result = Artisan::output();
|
||||||
|
if (str_contains($result, 'Encryption keys already exist')) {
|
||||||
|
$this->friendlyInfo('Encryption keys exist already.');
|
||||||
|
|
||||||
|
return CommandAlias::SUCCESS;
|
||||||
|
}
|
||||||
|
$this->friendlyPositive('Encryption keys have been created, nice!');
|
||||||
|
|
||||||
|
return CommandAlias::SUCCESS;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
<?php
|
<?php
|
||||||
/**
|
/*
|
||||||
* ApplyRules.php
|
* ApplyRules.php
|
||||||
* Copyright (c) 2020 james@firefly-iii.org
|
* Copyright (c) 2024 james@firefly-iii.org.
|
||||||
*
|
*
|
||||||
* This file is part of Firefly III (https://github.com/firefly-iii).
|
* This file is part of Firefly III (https://github.com/firefly-iii).
|
||||||
*
|
*
|
||||||
@@ -16,7 +16,7 @@
|
|||||||
* GNU Affero General Public License for more details.
|
* GNU Affero General Public License for more details.
|
||||||
*
|
*
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
* 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/>.
|
* along with this program. If not, see https://www.gnu.org/licenses/.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
/*
|
||||||
* Cron.php
|
* Cron.php
|
||||||
* Copyright (c) 2020 james@firefly-iii.org
|
* Copyright (c) 2024 james@firefly-iii.org.
|
||||||
*
|
*
|
||||||
* This file is part of Firefly III (https://github.com/firefly-iii).
|
* This file is part of Firefly III (https://github.com/firefly-iii).
|
||||||
*
|
*
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
* GNU Affero General Public License for more details.
|
* GNU Affero General Public License for more details.
|
||||||
*
|
*
|
||||||
* You should have received a copy of the GNU Affero General Public License
|
* 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/>.
|
* along with this program. If not, see https://www.gnu.org/licenses/.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
|
|||||||
@@ -110,7 +110,7 @@ class AppendBudgetLimitPeriods extends Command
|
|||||||
return 'daily';
|
return 'daily';
|
||||||
}
|
}
|
||||||
// is weekly
|
// is weekly
|
||||||
if ('1' === $limit->start_date->format('N') && '7' === $limit->end_date->format('N') && 6 === $limit->end_date->diffInDays($limit->start_date)) {
|
if ('1' === $limit->start_date->format('N') && '7' === $limit->end_date->format('N') && 6 === (int)$limit->end_date->diffInDays($limit->start_date, true)) {
|
||||||
return 'weekly';
|
return 'weekly';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -129,7 +129,7 @@ class AppendBudgetLimitPeriods extends Command
|
|||||||
if (
|
if (
|
||||||
in_array($limit->start_date->format('j-n'), $start, true) // start of quarter
|
in_array($limit->start_date->format('j-n'), $start, true) // start of quarter
|
||||||
&& in_array($limit->end_date->format('j-n'), $end, true) // end of quarter
|
&& in_array($limit->end_date->format('j-n'), $end, true) // end of quarter
|
||||||
&& 2 === $limit->start_date->diffInMonths($limit->end_date)
|
&& 2 === (int)$limit->start_date->diffInMonths($limit->end_date, true)
|
||||||
) {
|
) {
|
||||||
return 'quarterly';
|
return 'quarterly';
|
||||||
}
|
}
|
||||||
@@ -139,7 +139,7 @@ class AppendBudgetLimitPeriods extends Command
|
|||||||
if (
|
if (
|
||||||
in_array($limit->start_date->format('j-n'), $start, true) // start of quarter
|
in_array($limit->start_date->format('j-n'), $start, true) // start of quarter
|
||||||
&& in_array($limit->end_date->format('j-n'), $end, true) // end of quarter
|
&& in_array($limit->end_date->format('j-n'), $end, true) // end of quarter
|
||||||
&& 5 === $limit->start_date->diffInMonths($limit->end_date)
|
&& 5 === (int)$limit->start_date->diffInMonths($limit->end_date, true)
|
||||||
) {
|
) {
|
||||||
return 'half_year';
|
return 'half_year';
|
||||||
}
|
}
|
||||||
|
|||||||
182
app/Console/Commands/Upgrade/MigrateRuleActions.php
Normal file
182
app/Console/Commands/Upgrade/MigrateRuleActions.php
Normal file
@@ -0,0 +1,182 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
/*
|
||||||
|
* MigrateRuleActions.php
|
||||||
|
* Copyright (c) 2024 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\Console\Commands\Upgrade;
|
||||||
|
|
||||||
|
use FireflyIII\Console\Commands\ShowsFriendlyMessages;
|
||||||
|
use FireflyIII\Models\RuleAction;
|
||||||
|
use Illuminate\Console\Command;
|
||||||
|
|
||||||
|
class MigrateRuleActions extends Command
|
||||||
|
{
|
||||||
|
use ShowsFriendlyMessages;
|
||||||
|
|
||||||
|
public const string CONFIG_NAME = '610_migrate_rule_actions';
|
||||||
|
|
||||||
|
protected $description = 'Migrate rule actions away from expression engine';
|
||||||
|
|
||||||
|
protected $signature = 'firefly-iii:migrate-rule-actions {--F|force : Force the execution of this command.}';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Execute the console command.
|
||||||
|
*/
|
||||||
|
public function handle(): int
|
||||||
|
{
|
||||||
|
if ($this->isExecuted() && true !== $this->option('force')) {
|
||||||
|
$this->friendlyInfo('This command has already been executed.');
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
if (false === config('firefly.feature_flags.expression_engine')) {
|
||||||
|
$this->friendlyInfo('Expression engine is not enabled. Nothing to do.');
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
$this->replaceEqualSign();
|
||||||
|
$this->replaceObsoleteActions();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function isExecuted(): bool
|
||||||
|
{
|
||||||
|
$configVar = app('fireflyconfig')->get(self::CONFIG_NAME, false);
|
||||||
|
if (null !== $configVar) {
|
||||||
|
return (bool)$configVar->data;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function replaceEqualSign(): void
|
||||||
|
{
|
||||||
|
$count = 0;
|
||||||
|
$actions = RuleAction::get();
|
||||||
|
|
||||||
|
/** @var RuleAction $action */
|
||||||
|
foreach ($actions as $action) {
|
||||||
|
if (str_starts_with($action->action_value, '=')) {
|
||||||
|
$action->action_value = sprintf('%s%s', '\=', substr($action->action_value, 1));
|
||||||
|
$action->save();
|
||||||
|
++$count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($count > 0) {
|
||||||
|
$this->friendlyInfo(sprintf('Upgrading %d rule action(s) for the new expression engine.', $count));
|
||||||
|
}
|
||||||
|
if (0 === $count) {
|
||||||
|
$this->friendlyInfo('All rule actions are up to date.');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function replaceObsoleteActions(): void
|
||||||
|
{
|
||||||
|
$obsolete = [
|
||||||
|
'append_description',
|
||||||
|
'prepend_description',
|
||||||
|
'append_notes',
|
||||||
|
'prepend_notes',
|
||||||
|
'append_descr_to_notes',
|
||||||
|
'append_notes_to_descr',
|
||||||
|
'move_descr_to_notes',
|
||||||
|
'move_notes_to_descr',
|
||||||
|
];
|
||||||
|
$actions = RuleAction::whereIn('action_type', $obsolete)->get();
|
||||||
|
|
||||||
|
/** @var RuleAction $action */
|
||||||
|
foreach ($actions as $action) {
|
||||||
|
$oldType = $action->action_type;
|
||||||
|
|
||||||
|
switch ($action->action_type) {
|
||||||
|
default:
|
||||||
|
$this->friendlyError(sprintf('Cannot deal with action type "%s", skip it.', $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'append_description':
|
||||||
|
$action->action_type = 'set_description';
|
||||||
|
$action->action_value = sprintf('=description~"%s"', str_replace('"', '\"', $action->action_value));
|
||||||
|
$action->save();
|
||||||
|
$this->friendlyInfo(sprintf('Upgraded action #%d from "%s" to "%s".', $action->id, $oldType, $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'prepend_description':
|
||||||
|
$action->action_type = 'set_description';
|
||||||
|
$action->action_value = sprintf('="%s"~description', str_replace('"', '\"', $action->action_value));
|
||||||
|
$action->save();
|
||||||
|
$this->friendlyInfo(sprintf('Upgraded action #%d from "%s" to "%s".', $action->id, $oldType, $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'append_notes':
|
||||||
|
$action->action_type = 'set_notes';
|
||||||
|
$action->action_value = sprintf('=notes~"%s"', str_replace('"', '\"', $action->action_value));
|
||||||
|
$action->save();
|
||||||
|
$this->friendlyInfo(sprintf('Upgraded action #%d from "%s" to "%s".', $action->id, $oldType, $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'prepend_notes':
|
||||||
|
$action->action_type = 'set_notes';
|
||||||
|
$action->action_value = sprintf('="%s"~notes', str_replace('"', '\"', $action->action_value));
|
||||||
|
$action->save();
|
||||||
|
$this->friendlyInfo(sprintf('Upgraded action #%d from "%s" to "%s".', $action->id, $oldType, $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'append_descr_to_notes':
|
||||||
|
$action->action_type = 'set_notes';
|
||||||
|
$action->action_value = '=notes~" "~description';
|
||||||
|
$action->save();
|
||||||
|
$this->friendlyInfo(sprintf('Upgraded action #%d from "%s" to "%s".', $action->id, $oldType, $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'append_notes_to_descr':
|
||||||
|
$action->action_type = 'set_description';
|
||||||
|
$action->action_value = '=description~" "~notes';
|
||||||
|
$action->save();
|
||||||
|
$this->friendlyInfo(sprintf('Upgraded action #%d from "%s" to "%s".', $action->id, $oldType, $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'move_descr_to_notes':
|
||||||
|
$action->action_type = 'set_notes';
|
||||||
|
$action->action_value = '=description';
|
||||||
|
$action->save();
|
||||||
|
$this->friendlyInfo(sprintf('Upgraded action #%d from "%s" to "%s".', $action->id, $oldType, $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'move_notes_to_descr':
|
||||||
|
$action->action_type = 'set_description';
|
||||||
|
$action->action_value = '=notes';
|
||||||
|
$action->save();
|
||||||
|
$this->friendlyInfo(sprintf('Upgraded action #%d from "%s" to "%s".', $action->id, $oldType, $action->action_type));
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -63,6 +63,7 @@ class UpgradeDatabase extends Command
|
|||||||
'firefly-iii:upgrade-liabilities',
|
'firefly-iii:upgrade-liabilities',
|
||||||
'firefly-iii:liabilities-600',
|
'firefly-iii:liabilities-600',
|
||||||
'firefly-iii:budget-limit-periods',
|
'firefly-iii:budget-limit-periods',
|
||||||
|
'firefly-iii:migrate-rule-actions',
|
||||||
'firefly-iii:restore-oauth-keys',
|
'firefly-iii:restore-oauth-keys',
|
||||||
// also just in case, some integrity commands:
|
// also just in case, some integrity commands:
|
||||||
'firefly-iii:create-group-memberships',
|
'firefly-iii:create-group-memberships',
|
||||||
|
|||||||
@@ -26,11 +26,11 @@ class UpgradeSkeleton extends Command
|
|||||||
{
|
{
|
||||||
$start = microtime(true);
|
$start = microtime(true);
|
||||||
if ($this->isExecuted() && true !== $this->option('force')) {
|
if ($this->isExecuted() && true !== $this->option('force')) {
|
||||||
$this->info('FRIENDLY This command has already been executed.');
|
$this->friendlyInfo('This command has already been executed.');
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
$this->warn('Congrats, you found the skeleton command. Boo!');
|
$this->friendlyWarning('Congrats, you found the skeleton command. Boo!');
|
||||||
|
|
||||||
//$this->markAsExecuted();
|
//$this->markAsExecuted();
|
||||||
|
|
||||||
|
|||||||
@@ -40,12 +40,12 @@ class ReportGeneratorFactory
|
|||||||
{
|
{
|
||||||
$period = 'Month';
|
$period = 'Month';
|
||||||
// more than two months date difference means year report.
|
// more than two months date difference means year report.
|
||||||
if ($start->diffInMonths($end) > 1) {
|
if ($start->diffInMonths($end, true) > 1) {
|
||||||
$period = 'Year';
|
$period = 'Year';
|
||||||
}
|
}
|
||||||
|
|
||||||
// more than one year date difference means multi year report.
|
// more than one year date difference means multi-year report.
|
||||||
if ($start->diffInMonths($end) > 12) {
|
if ($start->diffInMonths($end, true) > 12) {
|
||||||
$period = 'MultiYear';
|
$period = 'MultiYear';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -405,7 +405,7 @@ class UserEventHandler
|
|||||||
}
|
}
|
||||||
// clean up old entries (6 months)
|
// clean up old entries (6 months)
|
||||||
$carbon = Carbon::createFromFormat('Y-m-d H:i:s', $preference[$index]['time']);
|
$carbon = Carbon::createFromFormat('Y-m-d H:i:s', $preference[$index]['time']);
|
||||||
if (false !== $carbon && $carbon->diffInMonths(today()) > 6) {
|
if (false !== $carbon && $carbon->diffInMonths(today(), true) > 6) {
|
||||||
app('log')->debug(sprintf('Entry for %s is very old, remove it.', $row['ip']));
|
app('log')->debug(sprintf('Entry for %s is very old, remove it.', $row['ip']));
|
||||||
unset($preference[$index]);
|
unset($preference[$index]);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -236,7 +236,9 @@ class AttachmentHelper implements AttachmentHelperInterface
|
|||||||
$fileObject->rewind();
|
$fileObject->rewind();
|
||||||
|
|
||||||
if (0 === $file->getSize()) {
|
if (0 === $file->getSize()) {
|
||||||
throw new FireflyException('Cannot upload empty or non-existent file.');
|
$this->errors->add('attachments', trans('validation.file_zero_length'));
|
||||||
|
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
$content = (string)$fileObject->fread($file->getSize());
|
$content = (string)$fileObject->fread($file->getSize());
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ class ReconcileController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
* */
|
* */
|
||||||
public function reconcile(Account $account, Carbon $start = null, Carbon $end = null)
|
public function reconcile(Account $account, ?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
if (!$this->isEditableAccount($account)) {
|
if (!$this->isEditableAccount($account)) {
|
||||||
return $this->redirectAccountToAccount($account);
|
return $this->redirectAccountToAccount($account);
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ class ShowController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
* */
|
* */
|
||||||
public function show(Request $request, Account $account, Carbon $start = null, Carbon $end = null)
|
public function show(Request $request, Account $account, ?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
$objectType = config(sprintf('firefly.shortNamesByFullName.%s', $account->accountType->type));
|
$objectType = config(sprintf('firefly.shortNamesByFullName.%s', $account->accountType->type));
|
||||||
|
|
||||||
|
|||||||
@@ -80,7 +80,18 @@ class LoginController extends Controller
|
|||||||
Log::channel('audit')->info(sprintf('User is trying to login using "%s"', $request->get($this->username())));
|
Log::channel('audit')->info(sprintf('User is trying to login using "%s"', $request->get($this->username())));
|
||||||
app('log')->debug('User is trying to login.');
|
app('log')->debug('User is trying to login.');
|
||||||
|
|
||||||
|
try {
|
||||||
$this->validateLogin($request);
|
$this->validateLogin($request);
|
||||||
|
} catch (ValidationException $e) {
|
||||||
|
return redirect(route('login'))
|
||||||
|
->withErrors(
|
||||||
|
[
|
||||||
|
$this->username => trans('auth.failed'),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
->onlyInput($this->username)
|
||||||
|
;
|
||||||
|
}
|
||||||
app('log')->debug('Login data is present.');
|
app('log')->debug('Login data is present.');
|
||||||
|
|
||||||
// Copied directly from AuthenticatesUsers, but with logging added:
|
// Copied directly from AuthenticatesUsers, but with logging added:
|
||||||
@@ -91,7 +102,6 @@ class LoginController extends Controller
|
|||||||
Log::channel('audit')->warning(sprintf('Login for user "%s" was locked out.', $request->get($this->username())));
|
Log::channel('audit')->warning(sprintf('Login for user "%s" was locked out.', $request->get($this->username())));
|
||||||
app('log')->error(sprintf('Login for user "%s" was locked out.', $request->get($this->username())));
|
app('log')->error(sprintf('Login for user "%s" was locked out.', $request->get($this->username())));
|
||||||
$this->fireLockoutEvent($request);
|
$this->fireLockoutEvent($request);
|
||||||
|
|
||||||
$this->sendLockoutResponse($request);
|
$this->sendLockoutResponse($request);
|
||||||
}
|
}
|
||||||
// Copied directly from AuthenticatesUsers, but with logging added:
|
// Copied directly from AuthenticatesUsers, but with logging added:
|
||||||
|
|||||||
@@ -85,7 +85,7 @@ class IndexController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
* */
|
* */
|
||||||
public function index(Carbon $start = null, Carbon $end = null)
|
public function index(?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
$this->abRepository->cleanup();
|
$this->abRepository->cleanup();
|
||||||
app('log')->debug(sprintf('Start of IndexController::index("%s", "%s")', $start?->format('Y-m-d'), $end?->format('Y-m-d')));
|
app('log')->debug(sprintf('Start of IndexController::index("%s", "%s")', $start?->format('Y-m-d'), $end?->format('Y-m-d')));
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ class ShowController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function noBudget(Request $request, Carbon $start = null, Carbon $end = null)
|
public function noBudget(Request $request, ?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
// @var Carbon $start
|
// @var Carbon $start
|
||||||
$start ??= session('start');
|
$start ??= session('start');
|
||||||
|
|||||||
@@ -70,7 +70,7 @@ class NoCategoryController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function show(Request $request, Carbon $start = null, Carbon $end = null)
|
public function show(Request $request, ?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
app('log')->debug('Start of noCategory()');
|
app('log')->debug('Start of noCategory()');
|
||||||
// @var Carbon $start
|
// @var Carbon $start
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ class ShowController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function show(Request $request, Category $category, Carbon $start = null, Carbon $end = null)
|
public function show(Request $request, Category $category, ?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
// @var Carbon $start
|
// @var Carbon $start
|
||||||
$start ??= session('start', today(config('app.timezone'))->startOfMonth());
|
$start ??= session('start', today(config('app.timezone'))->startOfMonth());
|
||||||
|
|||||||
@@ -143,7 +143,7 @@ class ReportController extends Controller
|
|||||||
$cache->addProperty($accounts);
|
$cache->addProperty($accounts);
|
||||||
$cache->addProperty($end);
|
$cache->addProperty($end);
|
||||||
if ($cache->has()) {
|
if ($cache->has()) {
|
||||||
return response()->json($cache->get());
|
// return response()->json($cache->get());
|
||||||
}
|
}
|
||||||
|
|
||||||
app('log')->debug('Going to do operations for accounts ', $accounts->pluck('id')->toArray());
|
app('log')->debug('Going to do operations for accounts ', $accounts->pluck('id')->toArray());
|
||||||
@@ -222,8 +222,11 @@ class ReportController extends Controller
|
|||||||
while ($currentStart <= $currentEnd) {
|
while ($currentStart <= $currentEnd) {
|
||||||
$key = $currentStart->format($format);
|
$key = $currentStart->format($format);
|
||||||
$title = $currentStart->isoFormat($titleFormat);
|
$title = $currentStart->isoFormat($titleFormat);
|
||||||
|
// #8663 make sure the period exists in the data previously collected.
|
||||||
|
if (array_key_exists($key, $currency)) {
|
||||||
$income['entries'][$title] = app('steam')->bcround($currency[$key]['earned'] ?? '0', $currency['currency_decimal_places']);
|
$income['entries'][$title] = app('steam')->bcround($currency[$key]['earned'] ?? '0', $currency['currency_decimal_places']);
|
||||||
$expense['entries'][$title] = app('steam')->bcround($currency[$key]['spent'] ?? '0', $currency['currency_decimal_places']);
|
$expense['entries'][$title] = app('steam')->bcround($currency[$key]['spent'] ?? '0', $currency['currency_decimal_places']);
|
||||||
|
}
|
||||||
$currentStart = app('navigation')->addPeriod($currentStart, $preferredRange, 0);
|
$currentStart = app('navigation')->addPeriod($currentStart, $preferredRange, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -89,7 +89,7 @@ class HomeController extends Controller
|
|||||||
$label = $request->get('label');
|
$label = $request->get('label');
|
||||||
$isCustomRange = false;
|
$isCustomRange = false;
|
||||||
|
|
||||||
app('log')->debug('Received dateRange', ['start' => $stringStart, 'end' => $stringEnd, 'label' => $request->get('label')]);
|
app('log')->debug('dateRange: Received dateRange', ['start' => $stringStart, 'end' => $stringEnd, 'label' => $request->get('label')]);
|
||||||
// check if the label is "everything" or "Custom range" which will betray
|
// check if the label is "everything" or "Custom range" which will betray
|
||||||
// a possible problem with the budgets.
|
// a possible problem with the budgets.
|
||||||
if ($label === (string)trans('firefly.everything') || $label === (string)trans('firefly.customRange')) {
|
if ($label === (string)trans('firefly.everything') || $label === (string)trans('firefly.customRange')) {
|
||||||
@@ -97,10 +97,10 @@ class HomeController extends Controller
|
|||||||
app('log')->debug('Range is now marked as "custom".');
|
app('log')->debug('Range is now marked as "custom".');
|
||||||
}
|
}
|
||||||
|
|
||||||
$diff = $start->diffInDays($end) + 1;
|
$diff = $start->diffInDays($end, true) + 1;
|
||||||
|
|
||||||
if ($diff > 50) {
|
if ($diff > 366) {
|
||||||
$request->session()->flash('warning', (string)trans('firefly.warning_much_data', ['days' => $diff]));
|
$request->session()->flash('warning', (string)trans('firefly.warning_much_data', ['days' => (int)$diff]));
|
||||||
}
|
}
|
||||||
|
|
||||||
$request->session()->put('is_custom_range', $isCustomRange);
|
$request->session()->put('is_custom_range', $isCustomRange);
|
||||||
|
|||||||
@@ -113,7 +113,7 @@ class BoxController extends Controller
|
|||||||
$spentAmount = $spent[$currency->id]['sum'] ?? '0';
|
$spentAmount = $spent[$currency->id]['sum'] ?? '0';
|
||||||
app('log')->debug(sprintf('Spent for default currency for all budgets in this period: %s', $spentAmount));
|
app('log')->debug(sprintf('Spent for default currency for all budgets in this period: %s', $spentAmount));
|
||||||
|
|
||||||
$days = $today->between($start, $end) ? $today->diffInDays($start) + 1 : $end->diffInDays($start) + 1;
|
$days = (int)($today->between($start, $end) ? $today->diffInDays($start, true) + 1 : $end->diffInDays($start, true) + 1);
|
||||||
app('log')->debug(sprintf('Number of days left: %d', $days));
|
app('log')->debug(sprintf('Number of days left: %d', $days));
|
||||||
$spentPerDay = bcdiv($spentAmount, (string)$days);
|
$spentPerDay = bcdiv($spentAmount, (string)$days);
|
||||||
app('log')->debug(sprintf('Available to spend per day: %s', $spentPerDay));
|
app('log')->debug(sprintf('Available to spend per day: %s', $spentPerDay));
|
||||||
@@ -130,7 +130,7 @@ class BoxController extends Controller
|
|||||||
$boxTitle = (string)trans('firefly.left_to_spend');
|
$boxTitle = (string)trans('firefly.left_to_spend');
|
||||||
$activeDaysLeft = $this->activeDaysLeft($start, $end); // see method description.
|
$activeDaysLeft = $this->activeDaysLeft($start, $end); // see method description.
|
||||||
$display = 1; // not overspent
|
$display = 1; // not overspent
|
||||||
$leftPerDayAmount = bcdiv($leftToSpendAmount, (string)$activeDaysLeft);
|
$leftPerDayAmount = 0 === (int) $activeDaysLeft ? $leftToSpendAmount : bcdiv($leftToSpendAmount, (string)$activeDaysLeft);
|
||||||
app('log')->debug(sprintf('Left to spend per day is %s', $leftPerDayAmount));
|
app('log')->debug(sprintf('Left to spend per day is %s', $leftPerDayAmount));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ class IntroController extends Controller
|
|||||||
/**
|
/**
|
||||||
* Returns the introduction wizard for a page.
|
* Returns the introduction wizard for a page.
|
||||||
*/
|
*/
|
||||||
public function getIntroSteps(string $route, string $specificPage = null): JsonResponse
|
public function getIntroSteps(string $route, ?string $specificPage = null): JsonResponse
|
||||||
{
|
{
|
||||||
app('log')->debug(sprintf('getIntroSteps for route "%s" and page "%s"', $route, $specificPage));
|
app('log')->debug(sprintf('getIntroSteps for route "%s" and page "%s"', $route, $specificPage));
|
||||||
$specificPage ??= '';
|
$specificPage ??= '';
|
||||||
@@ -91,7 +91,7 @@ class IntroController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function postEnable(string $route, string $specialPage = null): JsonResponse
|
public function postEnable(string $route, ?string $specialPage = null): JsonResponse
|
||||||
{
|
{
|
||||||
$specialPage ??= '';
|
$specialPage ??= '';
|
||||||
$route = str_replace('.', '_', $route);
|
$route = str_replace('.', '_', $route);
|
||||||
@@ -111,7 +111,7 @@ class IntroController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function postFinished(string $route, string $specialPage = null): JsonResponse
|
public function postFinished(string $route, ?string $specialPage = null): JsonResponse
|
||||||
{
|
{
|
||||||
$specialPage ??= '';
|
$specialPage ??= '';
|
||||||
$key = 'shown_demo_'.$route;
|
$key = 'shown_demo_'.$route;
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ class ReconcileController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function overview(Request $request, Account $account = null, Carbon $start = null, Carbon $end = null): JsonResponse
|
public function overview(Request $request, ?Account $account = null, ?Carbon $start = null, ?Carbon $end = null): JsonResponse
|
||||||
{
|
{
|
||||||
$startBalance = $request->get('startBalance');
|
$startBalance = $request->get('startBalance');
|
||||||
$endBalance = $request->get('endBalance');
|
$endBalance = $request->get('endBalance');
|
||||||
@@ -177,7 +177,7 @@ class ReconcileController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function transactions(Account $account, Carbon $start = null, Carbon $end = null)
|
public function transactions(Account $account, ?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
if (null === $start || null === $end) {
|
if (null === $start || null === $end) {
|
||||||
throw new FireflyException('Invalid dates submitted.');
|
throw new FireflyException('Invalid dates submitted.');
|
||||||
|
|||||||
@@ -76,7 +76,7 @@ class CreateController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function create(Request $request, RuleGroup $ruleGroup = null)
|
public function create(Request $request, ?RuleGroup $ruleGroup = null)
|
||||||
{
|
{
|
||||||
$this->createDefaultRuleGroup();
|
$this->createDefaultRuleGroup();
|
||||||
$preFilled = [
|
$preFilled = [
|
||||||
|
|||||||
@@ -215,7 +215,7 @@ class TagController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function show(Request $request, Tag $tag, Carbon $start = null, Carbon $end = null)
|
public function show(Request $request, Tag $tag, ?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
// default values:
|
// default values:
|
||||||
$subTitleIcon = 'fa-tag';
|
$subTitleIcon = 'fa-tag';
|
||||||
@@ -312,6 +312,9 @@ class TagController extends Controller
|
|||||||
if (count($this->attachmentsHelper->getMessages()->get('attachments')) > 0) {
|
if (count($this->attachmentsHelper->getMessages()->get('attachments')) > 0) {
|
||||||
$request->session()->flash('info', $this->attachmentsHelper->getMessages()->get('attachments'));
|
$request->session()->flash('info', $this->attachmentsHelper->getMessages()->get('attachments'));
|
||||||
}
|
}
|
||||||
|
if (count($this->attachmentsHelper->getErrors()->get('attachments')) > 0) {
|
||||||
|
$request->session()->flash('error', $this->attachmentsHelper->getErrors()->get('attachments'));
|
||||||
|
}
|
||||||
$redirect = redirect($this->getPreviousUrl('tags.create.url'));
|
$redirect = redirect($this->getPreviousUrl('tags.create.url'));
|
||||||
if (1 === (int)$request->get('create_another')) {
|
if (1 === (int)$request->get('create_another')) {
|
||||||
session()->put('tags.create.fromStore', true);
|
session()->put('tags.create.fromStore', true);
|
||||||
@@ -347,6 +350,9 @@ class TagController extends Controller
|
|||||||
if (count($this->attachmentsHelper->getMessages()->get('attachments')) > 0) {
|
if (count($this->attachmentsHelper->getMessages()->get('attachments')) > 0) {
|
||||||
$request->session()->flash('info', $this->attachmentsHelper->getMessages()->get('attachments'));
|
$request->session()->flash('info', $this->attachmentsHelper->getMessages()->get('attachments'));
|
||||||
}
|
}
|
||||||
|
if (count($this->attachmentsHelper->getErrors()->get('attachments')) > 0) {
|
||||||
|
$request->session()->flash('error', $this->attachmentsHelper->getErrors()->get('attachments'));
|
||||||
|
}
|
||||||
$redirect = redirect($this->getPreviousUrl('tags.edit.url'));
|
$redirect = redirect($this->getPreviousUrl('tags.edit.url'));
|
||||||
if (1 === (int)$request->get('return_to_edit')) {
|
if (1 === (int)$request->get('return_to_edit')) {
|
||||||
session()->put('tags.edit.fromUpdate', true);
|
session()->put('tags.edit.fromUpdate', true);
|
||||||
|
|||||||
@@ -69,7 +69,7 @@ class IndexController extends Controller
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function index(Request $request, string $objectType, Carbon $start = null, Carbon $end = null)
|
public function index(Request $request, string $objectType, ?Carbon $start = null, ?Carbon $end = null)
|
||||||
{
|
{
|
||||||
if ('transfers' === $objectType) {
|
if ('transfers' === $objectType) {
|
||||||
$objectType = 'transfer';
|
$objectType = 'transfer';
|
||||||
|
|||||||
@@ -50,12 +50,12 @@ class SecureHeaders
|
|||||||
$csp = [
|
$csp = [
|
||||||
"default-src 'none'",
|
"default-src 'none'",
|
||||||
"object-src 'none'",
|
"object-src 'none'",
|
||||||
sprintf("script-src 'unsafe-eval' 'strict-dynamic' 'self' 'unsafe-inline' 'nonce-%1s' %2s", $nonce, $trackingScriptSrc),
|
sprintf("script-src 'unsafe-eval' 'strict-dynamic' 'nonce-%1s'", $nonce),
|
||||||
"style-src 'unsafe-inline' 'self'",
|
"style-src 'unsafe-inline' 'self'",
|
||||||
"base-uri 'self'",
|
"base-uri 'self'",
|
||||||
"font-src 'self' data:",
|
"font-src 'self' data:",
|
||||||
sprintf("connect-src 'self' %s", $trackingScriptSrc),
|
sprintf("connect-src 'self' %s", $trackingScriptSrc),
|
||||||
sprintf("img-src data: 'strict-dynamic' 'self' *.tile.openstreetmap.org %s", $trackingScriptSrc),
|
sprintf("img-src 'self' 'nonce-%1s'", $nonce),
|
||||||
"manifest-src 'self'",
|
"manifest-src 'self'",
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ declare(strict_types=1);
|
|||||||
namespace FireflyIII\Http\Requests;
|
namespace FireflyIII\Http\Requests;
|
||||||
|
|
||||||
use FireflyIII\Models\Rule;
|
use FireflyIII\Models\Rule;
|
||||||
|
use FireflyIII\Rules\IsValidActionExpression;
|
||||||
use FireflyIII\Support\Request\ChecksLogin;
|
use FireflyIII\Support\Request\ChecksLogin;
|
||||||
use FireflyIII\Support\Request\ConvertsDataTypes;
|
use FireflyIII\Support\Request\ConvertsDataTypes;
|
||||||
use FireflyIII\Support\Request\GetRuleConfiguration;
|
use FireflyIII\Support\Request\GetRuleConfiguration;
|
||||||
@@ -147,7 +148,7 @@ class RuleFormRequest extends FormRequest
|
|||||||
'triggers.*.type' => 'required|in:'.implode(',', $validTriggers),
|
'triggers.*.type' => 'required|in:'.implode(',', $validTriggers),
|
||||||
'triggers.*.value' => sprintf('required_if:triggers.*.type,%s|max:1024|min:1|ruleTriggerValue', $contextTriggers),
|
'triggers.*.value' => sprintf('required_if:triggers.*.type,%s|max:1024|min:1|ruleTriggerValue', $contextTriggers),
|
||||||
'actions.*.type' => 'required|in:'.implode(',', $validActions),
|
'actions.*.type' => 'required|in:'.implode(',', $validActions),
|
||||||
'actions.*.value' => sprintf('required_if:actions.*.type,%s|min:0|max:1024|ruleActionValue', $contextActions),
|
'actions.*.value' => [sprintf('required_if:actions.*.type,%s|min:0|max:1024', $contextActions), new IsValidActionExpression(), 'ruleActionValue'],
|
||||||
'strict' => 'in:0,1',
|
'strict' => 'in:0,1',
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|||||||
@@ -128,7 +128,7 @@ class WarnAboutBills implements ShouldQueue
|
|||||||
$today = clone $this->date;
|
$today = clone $this->date;
|
||||||
$carbon = clone $bill->{$field};
|
$carbon = clone $bill->{$field};
|
||||||
|
|
||||||
return $today->diffInDays($carbon, false);
|
return (int) $today->diffInDays($carbon);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function sendWarning(Bill $bill, string $field): void
|
private function sendWarning(Bill $bill, string $field): void
|
||||||
|
|||||||
@@ -274,6 +274,13 @@ class Account extends Model
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected function iban(): Attribute
|
||||||
|
{
|
||||||
|
return Attribute::make(
|
||||||
|
get: static fn ($value) => null === $value ? null : trim(str_replace(' ', '', (string)$value)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
protected function order(): Attribute
|
protected function order(): Attribute
|
||||||
{
|
{
|
||||||
return Attribute::make(
|
return Attribute::make(
|
||||||
|
|||||||
@@ -26,10 +26,13 @@ namespace FireflyIII\Models;
|
|||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use Eloquent;
|
use Eloquent;
|
||||||
use FireflyIII\Support\Models\ReturnsIntegerIdTrait;
|
use FireflyIII\Support\Models\ReturnsIntegerIdTrait;
|
||||||
|
use FireflyIII\TransactionRules\Expressions\ActionExpression;
|
||||||
use Illuminate\Database\Eloquent\Builder;
|
use Illuminate\Database\Eloquent\Builder;
|
||||||
use Illuminate\Database\Eloquent\Casts\Attribute;
|
use Illuminate\Database\Eloquent\Casts\Attribute;
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
use Illuminate\Database\Eloquent\Relations\BelongsTo;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
use Symfony\Component\ExpressionLanguage\SyntaxError;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* FireflyIII\Models\RuleAction
|
* FireflyIII\Models\RuleAction
|
||||||
@@ -75,6 +78,30 @@ class RuleAction extends Model
|
|||||||
|
|
||||||
protected $fillable = ['rule_id', 'action_type', 'action_value', 'order', 'active', 'stop_processing'];
|
protected $fillable = ['rule_id', 'action_type', 'action_value', 'order', 'active', 'stop_processing'];
|
||||||
|
|
||||||
|
public function getValue(array $journal): string
|
||||||
|
{
|
||||||
|
if (false === config('firefly.feature_flags.expression_engine')) {
|
||||||
|
Log::debug('Expression engine is disabled, returning action value as string.');
|
||||||
|
|
||||||
|
return (string)$this->action_value;
|
||||||
|
}
|
||||||
|
if (true === config('firefly.feature_flags.expression_engine') && str_starts_with($this->action_value, '\=')) {
|
||||||
|
// return literal string.
|
||||||
|
return substr($this->action_value, 1);
|
||||||
|
}
|
||||||
|
$expr = new ActionExpression($this->action_value);
|
||||||
|
|
||||||
|
try {
|
||||||
|
$result = $expr->evaluate($journal);
|
||||||
|
} catch (SyntaxError $e) {
|
||||||
|
Log::error(sprintf('Expression engine failed to evaluate expression "%s" with error "%s".', $this->action_value, $e->getMessage()));
|
||||||
|
$result = (string)$this->action_value;
|
||||||
|
}
|
||||||
|
Log::debug(sprintf('Expression engine is enabled, result of expression "%s" is "%s".', $this->action_value, $result));
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
public function rule(): BelongsTo
|
public function rule(): BelongsTo
|
||||||
{
|
{
|
||||||
return $this->belongsTo(Rule::class);
|
return $this->belongsTo(Rule::class);
|
||||||
|
|||||||
@@ -25,10 +25,9 @@ namespace FireflyIII\Providers;
|
|||||||
|
|
||||||
use Illuminate\Support\Facades\Blade;
|
use Illuminate\Support\Facades\Blade;
|
||||||
use Illuminate\Support\Facades\Response;
|
use Illuminate\Support\Facades\Response;
|
||||||
|
use Illuminate\Support\Facades\Route;
|
||||||
use Illuminate\Support\Facades\Schema;
|
use Illuminate\Support\Facades\Schema;
|
||||||
use Illuminate\Support\ServiceProvider;
|
use Illuminate\Support\ServiceProvider;
|
||||||
use Laravel\Passport\Passport;
|
|
||||||
use Laravel\Sanctum\Sanctum;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class AppServiceProvider
|
* Class AppServiceProvider
|
||||||
@@ -65,9 +64,14 @@ class AppServiceProvider extends ServiceProvider
|
|||||||
|
|
||||||
return '';
|
return '';
|
||||||
});
|
});
|
||||||
Blade::if('partialroute', function (string $route) {
|
Blade::if('partialroute', function (string $route, string $firstParam = '') {
|
||||||
$name = \Route::getCurrentRoute()->getName() ?? '';
|
$name = Route::getCurrentRoute()->getName() ?? '';
|
||||||
if (str_contains($name, $route)) {
|
if ('' === $firstParam && str_contains($name, $route)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
$params = Route::getCurrentRoute()->parameters() ?? [];
|
||||||
|
$objectType = $params['objectType'] ?? '';
|
||||||
|
if ($objectType === $firstParam && str_contains($name, $route)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -80,7 +84,7 @@ class AppServiceProvider extends ServiceProvider
|
|||||||
*/
|
*/
|
||||||
public function register(): void
|
public function register(): void
|
||||||
{
|
{
|
||||||
Passport::ignoreMigrations();
|
// Passport::ignoreMigrations();
|
||||||
Sanctum::ignoreMigrations();
|
// Sanctum::ignoreMigrations();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -69,9 +69,11 @@ use FireflyIII\Support\Preferences;
|
|||||||
use FireflyIII\Support\Steam;
|
use FireflyIII\Support\Steam;
|
||||||
use FireflyIII\TransactionRules\Engine\RuleEngineInterface;
|
use FireflyIII\TransactionRules\Engine\RuleEngineInterface;
|
||||||
use FireflyIII\TransactionRules\Engine\SearchRuleEngine;
|
use FireflyIII\TransactionRules\Engine\SearchRuleEngine;
|
||||||
|
use FireflyIII\TransactionRules\Expressions\ActionExpressionLanguageProvider;
|
||||||
use FireflyIII\Validation\FireflyValidator;
|
use FireflyIII\Validation\FireflyValidator;
|
||||||
use Illuminate\Foundation\Application;
|
use Illuminate\Foundation\Application;
|
||||||
use Illuminate\Support\ServiceProvider;
|
use Illuminate\Support\ServiceProvider;
|
||||||
|
use Symfony\Component\ExpressionLanguage\ExpressionLanguage;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class FireflyServiceProvider.
|
* Class FireflyServiceProvider.
|
||||||
@@ -200,6 +202,17 @@ class FireflyServiceProvider extends ServiceProvider
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// rule expression language
|
||||||
|
$this->app->singleton(
|
||||||
|
ExpressionLanguage::class,
|
||||||
|
static function () {
|
||||||
|
$expressionLanguage = new ExpressionLanguage();
|
||||||
|
$expressionLanguage->registerProvider(new ActionExpressionLanguageProvider());
|
||||||
|
|
||||||
|
return $expressionLanguage;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
$this->app->bind(
|
$this->app->bind(
|
||||||
RuleEngineInterface::class,
|
RuleEngineInterface::class,
|
||||||
static function (Application $app) {
|
static function (Application $app) {
|
||||||
|
|||||||
@@ -122,7 +122,7 @@ class BudgetLimitRepository implements BudgetLimitRepositoryInterface
|
|||||||
$budgetLimit->delete();
|
$budgetLimit->delete();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getAllBudgetLimitsByCurrency(TransactionCurrency $currency, Carbon $start = null, Carbon $end = null): Collection
|
public function getAllBudgetLimitsByCurrency(TransactionCurrency $currency, ?Carbon $start = null, ?Carbon $end = null): Collection
|
||||||
{
|
{
|
||||||
return $this->getAllBudgetLimits($start, $end)->filter(
|
return $this->getAllBudgetLimits($start, $end)->filter(
|
||||||
static function (BudgetLimit $budgetLimit) use ($currency) {
|
static function (BudgetLimit $budgetLimit) use ($currency) {
|
||||||
@@ -131,7 +131,7 @@ class BudgetLimitRepository implements BudgetLimitRepositoryInterface
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getAllBudgetLimits(Carbon $start = null, Carbon $end = null): Collection
|
public function getAllBudgetLimits(?Carbon $start = null, ?Carbon $end = null): Collection
|
||||||
{
|
{
|
||||||
// both are NULL:
|
// both are NULL:
|
||||||
if (null === $start && null === $end) {
|
if (null === $start && null === $end) {
|
||||||
@@ -198,7 +198,7 @@ class BudgetLimitRepository implements BudgetLimitRepositoryInterface
|
|||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function getBudgetLimits(Budget $budget, Carbon $start = null, Carbon $end = null): Collection
|
public function getBudgetLimits(Budget $budget, ?Carbon $start = null, ?Carbon $end = null): Collection
|
||||||
{
|
{
|
||||||
if (null === $end && null === $start) {
|
if (null === $end && null === $start) {
|
||||||
return $budget->budgetlimits()->with(['transactionCurrency'])->orderBy('budget_limits.start_date', 'DESC')->get(['budget_limits.*']);
|
return $budget->budgetlimits()->with(['transactionCurrency'])->orderBy('budget_limits.start_date', 'DESC')->get(['budget_limits.*']);
|
||||||
|
|||||||
@@ -57,11 +57,11 @@ interface BudgetLimitRepositoryInterface
|
|||||||
/**
|
/**
|
||||||
* TODO this method is not multi currency aware.
|
* TODO this method is not multi currency aware.
|
||||||
*/
|
*/
|
||||||
public function getAllBudgetLimits(Carbon $start = null, Carbon $end = null): Collection;
|
public function getAllBudgetLimits(?Carbon $start = null, ?Carbon $end = null): Collection;
|
||||||
|
|
||||||
public function getAllBudgetLimitsByCurrency(TransactionCurrency $currency, Carbon $start = null, Carbon $end = null): Collection;
|
public function getAllBudgetLimitsByCurrency(TransactionCurrency $currency, ?Carbon $start = null, ?Carbon $end = null): Collection;
|
||||||
|
|
||||||
public function getBudgetLimits(Budget $budget, Carbon $start = null, Carbon $end = null): Collection;
|
public function getBudgetLimits(Budget $budget, ?Carbon $start = null, ?Carbon $end = null): Collection;
|
||||||
|
|
||||||
public function setUser(null|Authenticatable|User $user): void;
|
public function setUser(null|Authenticatable|User $user): void;
|
||||||
|
|
||||||
|
|||||||
@@ -132,7 +132,7 @@ class BudgetRepository implements BudgetRepositoryInterface
|
|||||||
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
$total = $limit->start_date->diffInDays($limit->end_date) + 1; // include the day itself.
|
$total = $limit->start_date->diffInDays($limit->end_date, true) + 1; // include the day itself.
|
||||||
$days = $this->daysInOverlap($limit, $start, $end);
|
$days = $this->daysInOverlap($limit, $start, $end);
|
||||||
$amount = bcmul(bcdiv($limit->amount, (string)$total), (string)$days);
|
$amount = bcmul(bcdiv($limit->amount, (string)$total), (string)$days);
|
||||||
$return[$currencyCode]['sum'] = bcadd($return[$currencyCode]['sum'], $amount);
|
$return[$currencyCode]['sum'] = bcadd($return[$currencyCode]['sum'], $amount);
|
||||||
@@ -183,21 +183,21 @@ class BudgetRepository implements BudgetRepositoryInterface
|
|||||||
// |-----------|
|
// |-----------|
|
||||||
// |----------------|
|
// |----------------|
|
||||||
if ($start->gte($limit->start_date) && $end->lte($limit->end_date)) {
|
if ($start->gte($limit->start_date) && $end->lte($limit->end_date)) {
|
||||||
return $start->diffInDays($end) + 1; // add one day
|
return (int) $start->diffInDays($end, true) + 1; // add one day
|
||||||
}
|
}
|
||||||
// limit starts earlier and limit ends first:
|
// limit starts earlier and limit ends first:
|
||||||
// |-----------|
|
// |-----------|
|
||||||
// |-------|
|
// |-------|
|
||||||
if ($limit->start_date->lte($start) && $limit->end_date->lte($end)) {
|
if ($limit->start_date->lte($start) && $limit->end_date->lte($end)) {
|
||||||
// return days in the range $start-$limit_end
|
// return days in the range $start-$limit_end
|
||||||
return $start->diffInDays($limit->end_date) + 1; // add one day, the day itself
|
return (int) $start->diffInDays($limit->end_date, true) + 1; // add one day, the day itself
|
||||||
}
|
}
|
||||||
// limit starts later and limit ends earlier
|
// limit starts later and limit ends earlier
|
||||||
// |-----------|
|
// |-----------|
|
||||||
// |-------|
|
// |-------|
|
||||||
if ($limit->start_date->gte($start) && $limit->end_date->gte($end)) {
|
if ($limit->start_date->gte($start) && $limit->end_date->gte($end)) {
|
||||||
// return days in the range $limit_start - $end
|
// return days in the range $limit_start - $end
|
||||||
return $limit->start_date->diffInDays($end) + 1; // add one day, the day itself
|
return (int) $limit->start_date->diffInDays($end, true) + 1; // add one day, the day itself
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@@ -438,7 +438,7 @@ class BudgetRepository implements BudgetRepositoryInterface
|
|||||||
*
|
*
|
||||||
* @param null|int $budgetId |null
|
* @param null|int $budgetId |null
|
||||||
*/
|
*/
|
||||||
public function find(int $budgetId = null): ?Budget
|
public function find(?int $budgetId = null): ?Budget
|
||||||
{
|
{
|
||||||
return $this->user->budgets()->find($budgetId);
|
return $this->user->budgets()->find($budgetId);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ interface BudgetRepositoryInterface
|
|||||||
|
|
||||||
public function destroyAutoBudget(Budget $budget): void;
|
public function destroyAutoBudget(Budget $budget): void;
|
||||||
|
|
||||||
public function find(int $budgetId = null): ?Budget;
|
public function find(?int $budgetId = null): ?Budget;
|
||||||
|
|
||||||
public function findBudget(?int $budgetId, ?string $budgetName): ?Budget;
|
public function findBudget(?int $budgetId, ?string $budgetName): ?Budget;
|
||||||
|
|
||||||
|
|||||||
@@ -51,7 +51,7 @@ class OperationsRepository implements OperationsRepositoryInterface
|
|||||||
$total = '0';
|
$total = '0';
|
||||||
$count = 0;
|
$count = 0;
|
||||||
foreach ($budget->budgetlimits as $limit) {
|
foreach ($budget->budgetlimits as $limit) {
|
||||||
$diff = $limit->start_date->diffInDays($limit->end_date);
|
$diff = (int) $limit->start_date->diffInDays($limit->end_date, true);
|
||||||
$diff = 0 === $diff ? 1 : $diff;
|
$diff = 0 === $diff ? 1 : $diff;
|
||||||
$amount = $limit->amount;
|
$amount = $limit->amount;
|
||||||
$perDay = bcdiv($amount, (string)$diff);
|
$perDay = bcdiv($amount, (string)$diff);
|
||||||
|
|||||||
@@ -44,7 +44,7 @@ class LinkTypeRepository implements LinkTypeRepositoryInterface
|
|||||||
return $linkType->transactionJournalLinks()->count();
|
return $linkType->transactionJournalLinks()->count();
|
||||||
}
|
}
|
||||||
|
|
||||||
public function destroy(LinkType $linkType, LinkType $moveTo = null): bool
|
public function destroy(LinkType $linkType, ?LinkType $moveTo = null): bool
|
||||||
{
|
{
|
||||||
if (null !== $moveTo) {
|
if (null !== $moveTo) {
|
||||||
TransactionJournalLink::where('link_type_id', $linkType->id)->update(['link_type_id' => $moveTo->id]);
|
TransactionJournalLink::where('link_type_id', $linkType->id)->update(['link_type_id' => $moveTo->id]);
|
||||||
@@ -113,7 +113,7 @@ class LinkTypeRepository implements LinkTypeRepositoryInterface
|
|||||||
/**
|
/**
|
||||||
* Returns all the journal links (of a specific type).
|
* Returns all the journal links (of a specific type).
|
||||||
*/
|
*/
|
||||||
public function getJournalLinks(LinkType $linkType = null): Collection
|
public function getJournalLinks(?LinkType $linkType = null): Collection
|
||||||
{
|
{
|
||||||
$query = TransactionJournalLink::with(['source', 'destination'])
|
$query = TransactionJournalLink::with(['source', 'destination'])
|
||||||
->leftJoin('transaction_journals as source_journals', 'journal_links.source_id', '=', 'source_journals.id')
|
->leftJoin('transaction_journals as source_journals', 'journal_links.source_id', '=', 'source_journals.id')
|
||||||
@@ -225,7 +225,7 @@ class LinkTypeRepository implements LinkTypeRepositoryInterface
|
|||||||
return LinkType::find($linkTypeId);
|
return LinkType::find($linkTypeId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function findByName(string $name = null): ?LinkType
|
public function findByName(?string $name = null): ?LinkType
|
||||||
{
|
{
|
||||||
if (null === $name) {
|
if (null === $name) {
|
||||||
return null;
|
return null;
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ interface LinkTypeRepositoryInterface
|
|||||||
{
|
{
|
||||||
public function countJournals(LinkType $linkType): int;
|
public function countJournals(LinkType $linkType): int;
|
||||||
|
|
||||||
public function destroy(LinkType $linkType, LinkType $moveTo = null): bool;
|
public function destroy(LinkType $linkType, ?LinkType $moveTo = null): bool;
|
||||||
|
|
||||||
public function destroyLink(TransactionJournalLink $link): bool;
|
public function destroyLink(TransactionJournalLink $link): bool;
|
||||||
|
|
||||||
@@ -46,7 +46,7 @@ interface LinkTypeRepositoryInterface
|
|||||||
/**
|
/**
|
||||||
* Find link type by name.
|
* Find link type by name.
|
||||||
*/
|
*/
|
||||||
public function findByName(string $name = null): ?LinkType;
|
public function findByName(?string $name = null): ?LinkType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if link exists between journals.
|
* Check if link exists between journals.
|
||||||
@@ -65,7 +65,7 @@ interface LinkTypeRepositoryInterface
|
|||||||
*/
|
*/
|
||||||
public function getJournalIds(LinkType $linkType): array;
|
public function getJournalIds(LinkType $linkType): array;
|
||||||
|
|
||||||
public function getJournalLinks(LinkType $linkType = null): Collection;
|
public function getJournalLinks(?LinkType $linkType = null): Collection;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return list of existing connections.
|
* Return list of existing connections.
|
||||||
|
|||||||
@@ -301,7 +301,7 @@ class PiggyBankRepository implements PiggyBankRepositoryInterface
|
|||||||
if (null !== $piggyBank->targetdate && $repetition->currentamount < $piggyBank->targetamount) {
|
if (null !== $piggyBank->targetdate && $repetition->currentamount < $piggyBank->targetamount) {
|
||||||
$now = today(config('app.timezone'));
|
$now = today(config('app.timezone'));
|
||||||
$startDate = null !== $piggyBank->startdate && $piggyBank->startdate->gte($now) ? $piggyBank->startdate : $now;
|
$startDate = null !== $piggyBank->startdate && $piggyBank->startdate->gte($now) ? $piggyBank->startdate : $now;
|
||||||
$diffInMonths = $startDate->diffInMonths($piggyBank->targetdate, false);
|
$diffInMonths = (int) $startDate->diffInMonths($piggyBank->targetdate);
|
||||||
$remainingAmount = bcsub($piggyBank->targetamount, $repetition->currentamount);
|
$remainingAmount = bcsub($piggyBank->targetamount, $repetition->currentamount);
|
||||||
|
|
||||||
// more than 1 month to go and still need money to save:
|
// more than 1 month to go and still need money to save:
|
||||||
|
|||||||
@@ -202,7 +202,7 @@ class RecurringRepository implements RecurringRepositoryInterface
|
|||||||
/**
|
/**
|
||||||
* Returns the journals created for this recurrence, possibly limited by time.
|
* Returns the journals created for this recurrence, possibly limited by time.
|
||||||
*/
|
*/
|
||||||
public function getJournalCount(Recurrence $recurrence, Carbon $start = null, Carbon $end = null): int
|
public function getJournalCount(Recurrence $recurrence, ?Carbon $start = null, ?Carbon $end = null): int
|
||||||
{
|
{
|
||||||
$query = TransactionJournal::leftJoin('journal_meta', 'journal_meta.transaction_journal_id', '=', 'transaction_journals.id')
|
$query = TransactionJournal::leftJoin('journal_meta', 'journal_meta.transaction_journal_id', '=', 'transaction_journals.id')
|
||||||
->where('transaction_journals.user_id', $recurrence->user_id)
|
->where('transaction_journals.user_id', $recurrence->user_id)
|
||||||
@@ -476,7 +476,7 @@ class RecurringRepository implements RecurringRepositoryInterface
|
|||||||
if (false === $repDate) {
|
if (false === $repDate) {
|
||||||
$repDate = clone $today;
|
$repDate = clone $today;
|
||||||
}
|
}
|
||||||
$diffInYears = $today->diffInYears($repDate);
|
$diffInYears = (int) $today->diffInYears($repDate, true);
|
||||||
$repDate->addYears($diffInYears); // technically not necessary.
|
$repDate->addYears($diffInYears); // technically not necessary.
|
||||||
$string = $repDate->isoFormat((string)trans('config.month_and_day_no_year_js'));
|
$string = $repDate->isoFormat((string)trans('config.month_and_day_no_year_js'));
|
||||||
|
|
||||||
|
|||||||
@@ -83,7 +83,7 @@ interface RecurringRepositoryInterface
|
|||||||
/**
|
/**
|
||||||
* Returns the count of journals created for this recurrence, possibly limited by time.
|
* Returns the count of journals created for this recurrence, possibly limited by time.
|
||||||
*/
|
*/
|
||||||
public function getJournalCount(Recurrence $recurrence, Carbon $start = null, Carbon $end = null): int;
|
public function getJournalCount(Recurrence $recurrence, ?Carbon $start = null, ?Carbon $end = null): int;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get journal ID's for journals created by this recurring transaction.
|
* Get journal ID's for journals created by this recurring transaction.
|
||||||
|
|||||||
@@ -28,6 +28,7 @@ use FireflyIII\Models\Account;
|
|||||||
use FireflyIII\Models\AccountMeta;
|
use FireflyIII\Models\AccountMeta;
|
||||||
use FireflyIII\Models\AccountType;
|
use FireflyIII\Models\AccountType;
|
||||||
use FireflyIII\Models\TransactionCurrency;
|
use FireflyIII\Models\TransactionCurrency;
|
||||||
|
use FireflyIII\Services\Internal\Update\AccountUpdateService;
|
||||||
use FireflyIII\Support\Repositories\UserGroup\UserGroupTrait;
|
use FireflyIII\Support\Repositories\UserGroup\UserGroupTrait;
|
||||||
use Illuminate\Database\Eloquent\Builder as EloquentBuilder;
|
use Illuminate\Database\Eloquent\Builder as EloquentBuilder;
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
@@ -239,6 +240,7 @@ class AccountRepository implements AccountRepositoryInterface
|
|||||||
|
|
||||||
public function getAccountsByType(array $types, ?array $sort = []): Collection
|
public function getAccountsByType(array $types, ?array $sort = []): Collection
|
||||||
{
|
{
|
||||||
|
$sortable = ['name', 'active']; // TODO yes this is a duplicate array.
|
||||||
$res = array_intersect([AccountType::ASSET, AccountType::MORTGAGE, AccountType::LOAN, AccountType::DEBT], $types);
|
$res = array_intersect([AccountType::ASSET, AccountType::MORTGAGE, AccountType::LOAN, AccountType::DEBT], $types);
|
||||||
$query = $this->userGroup->accounts();
|
$query = $this->userGroup->accounts();
|
||||||
if (0 !== count($types)) {
|
if (0 !== count($types)) {
|
||||||
@@ -246,17 +248,23 @@ class AccountRepository implements AccountRepositoryInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
// add sort parameters. At this point they're filtered to allowed fields to sort by:
|
// add sort parameters. At this point they're filtered to allowed fields to sort by:
|
||||||
if (0 !== count($sort)) {
|
$hasActiveColumn = array_key_exists('active', $sort);
|
||||||
foreach ($sort as $param) {
|
if (count($sort) > 0) {
|
||||||
$query->orderBy($param[0], $param[1]);
|
if (false === $hasActiveColumn) {
|
||||||
|
$query->orderBy('accounts.active', 'DESC');
|
||||||
|
}
|
||||||
|
foreach ($sort as $column => $direction) {
|
||||||
|
if (in_array($column, $sortable, true)) {
|
||||||
|
$query->orderBy(sprintf('accounts.%s', $column), $direction);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (0 === count($sort)) {
|
if (0 === count($sort)) {
|
||||||
if (0 !== count($res)) {
|
if (0 !== count($res)) {
|
||||||
$query->orderBy('accounts.order', 'ASC');
|
|
||||||
}
|
|
||||||
$query->orderBy('accounts.active', 'DESC');
|
$query->orderBy('accounts.active', 'DESC');
|
||||||
|
}
|
||||||
|
$query->orderBy('accounts.order', 'ASC');
|
||||||
$query->orderBy('accounts.name', 'ASC');
|
$query->orderBy('accounts.name', 'ASC');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -288,4 +296,13 @@ class AccountRepository implements AccountRepositoryInterface
|
|||||||
|
|
||||||
return $dbQuery->take($limit)->get(['accounts.*']);
|
return $dbQuery->take($limit)->get(['accounts.*']);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[\Override]
|
||||||
|
public function update(Account $account, array $data): Account
|
||||||
|
{
|
||||||
|
/** @var AccountUpdateService $service */
|
||||||
|
$service = app(AccountUpdateService::class);
|
||||||
|
|
||||||
|
return $service->update($account, $data);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -73,4 +73,6 @@ interface AccountRepositoryInterface
|
|||||||
public function setUser(User $user): void;
|
public function setUser(User $user): void;
|
||||||
|
|
||||||
public function setUserGroup(UserGroup $userGroup): void;
|
public function setUserGroup(UserGroup $userGroup): void;
|
||||||
|
|
||||||
|
public function update(Account $account, array $data): Account;
|
||||||
}
|
}
|
||||||
|
|||||||
57
app/Rules/IsValidActionExpression.php
Normal file
57
app/Rules/IsValidActionExpression.php
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
*
|
||||||
|
* IsValidActionExpression.php
|
||||||
|
* Copyright (c) 2024 Michael Thomas
|
||||||
|
*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace FireflyIII\Rules;
|
||||||
|
|
||||||
|
use FireflyIII\TransactionRules\Expressions\ActionExpression;
|
||||||
|
use Illuminate\Contracts\Validation\ValidationRule;
|
||||||
|
use Illuminate\Translation\PotentiallyTranslatedString;
|
||||||
|
|
||||||
|
class IsValidActionExpression implements ValidationRule
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Check that the given action expression is syntactically valid.
|
||||||
|
*
|
||||||
|
* @param \Closure(string): PotentiallyTranslatedString $fail
|
||||||
|
*
|
||||||
|
* @SuppressWarnings(PHPMD.UnusedFormalParameter)
|
||||||
|
*/
|
||||||
|
public function validate(string $attribute, mixed $value, \Closure $fail): void
|
||||||
|
{
|
||||||
|
if (false === config('firefly.feature_flags.expression_engine')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$value ??= '';
|
||||||
|
$expr = new ActionExpression($value);
|
||||||
|
|
||||||
|
if (!$expr->isValid()) {
|
||||||
|
$fail('validation.rule_action_expression')->translate(
|
||||||
|
[
|
||||||
|
'error' => $expr->getValidationError()->getMessage(),
|
||||||
|
]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -40,7 +40,7 @@ class Amount
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function formatAnything(TransactionCurrency $format, string $amount, bool $coloured = null): string
|
public function formatAnything(TransactionCurrency $format, string $amount, ?bool $coloured = null): string
|
||||||
{
|
{
|
||||||
return $this->formatFlat($format->symbol, $format->decimal_places, $amount, $coloured);
|
return $this->formatFlat($format->symbol, $format->decimal_places, $amount, $coloured);
|
||||||
}
|
}
|
||||||
@@ -53,7 +53,7 @@ class Amount
|
|||||||
*
|
*
|
||||||
* @SuppressWarnings(PHPMD.MissingImport)
|
* @SuppressWarnings(PHPMD.MissingImport)
|
||||||
*/
|
*/
|
||||||
public function formatFlat(string $symbol, int $decimalPlaces, string $amount, bool $coloured = null): string
|
public function formatFlat(string $symbol, int $decimalPlaces, string $amount, ?bool $coloured = null): string
|
||||||
{
|
{
|
||||||
$locale = app('steam')->getLocale();
|
$locale = app('steam')->getLocale();
|
||||||
$rounded = app('steam')->bcround($amount, $decimalPlaces);
|
$rounded = app('steam')->bcround($amount, $decimalPlaces);
|
||||||
|
|||||||
@@ -120,4 +120,12 @@ class RemoteUserProvider implements UserProvider
|
|||||||
|
|
||||||
throw new FireflyException(sprintf('C) Did not implement %s', __METHOD__));
|
throw new FireflyException(sprintf('C) Did not implement %s', __METHOD__));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[\Override]
|
||||||
|
public function rehashPasswordIfRequired(Authenticatable $user, array $credentials, bool $force = false): void
|
||||||
|
{
|
||||||
|
app('log')->debug(sprintf('Now at %s', __METHOD__));
|
||||||
|
|
||||||
|
throw new FireflyException(sprintf('Did not implement %s', __METHOD__));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -34,6 +34,10 @@ class EitherConfigKey
|
|||||||
{
|
{
|
||||||
public static array $static
|
public static array $static
|
||||||
= [
|
= [
|
||||||
|
// currency conversion
|
||||||
|
'cer.enabled',
|
||||||
|
|
||||||
|
// firefly iii settings
|
||||||
'firefly.version',
|
'firefly.version',
|
||||||
'firefly.api_version',
|
'firefly.api_version',
|
||||||
'firefly.default_location',
|
'firefly.default_location',
|
||||||
|
|||||||
@@ -113,7 +113,7 @@ class WholePeriodChartGenerator
|
|||||||
protected function calculateStep(Carbon $start, Carbon $end): string
|
protected function calculateStep(Carbon $start, Carbon $end): string
|
||||||
{
|
{
|
||||||
$step = '1D';
|
$step = '1D';
|
||||||
$months = $start->diffInMonths($end);
|
$months = $start->diffInMonths($end, true);
|
||||||
if ($months > 3) {
|
if ($months > 3) {
|
||||||
$step = '1W';
|
$step = '1W';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -40,7 +40,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function amountNoCurrency(string $name, $value = null, array $options = null): string
|
public function amountNoCurrency(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
@@ -71,7 +71,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function checkbox(string $name, int $value = null, $checked = null, array $options = null): string
|
public function checkbox(string $name, ?int $value = null, $checked = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
$value ??= 1;
|
$value ??= 1;
|
||||||
@@ -106,7 +106,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function date(string $name, $value = null, array $options = null): string
|
public function date(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
$options = $this->expandOptionArray($name, $label, $options);
|
$options = $this->expandOptionArray($name, $label, $options);
|
||||||
@@ -129,7 +129,7 @@ class ExpandedForm
|
|||||||
/**
|
/**
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function file(string $name, array $options = null): string
|
public function file(string $name, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
@@ -153,7 +153,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function integer(string $name, $value = null, array $options = null): string
|
public function integer(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
@@ -179,7 +179,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function location(string $name, $value = null, array $options = null): string
|
public function location(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
@@ -227,7 +227,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function objectGroup($value = null, array $options = null): string
|
public function objectGroup($value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$name = 'object_group';
|
$name = 'object_group';
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
@@ -272,7 +272,7 @@ class ExpandedForm
|
|||||||
/**
|
/**
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function password(string $name, array $options = null): string
|
public function password(string $name, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
$options = $this->expandOptionArray($name, $label, $options);
|
$options = $this->expandOptionArray($name, $label, $options);
|
||||||
@@ -297,7 +297,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function percentage(string $name, $value = null, array $options = null): string
|
public function percentage(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
$options = $this->expandOptionArray($name, $label, $options);
|
$options = $this->expandOptionArray($name, $label, $options);
|
||||||
@@ -323,7 +323,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function staticText(string $name, $value, array $options = null): string
|
public function staticText(string $name, $value, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
$options = $this->expandOptionArray($name, $label, $options);
|
$options = $this->expandOptionArray($name, $label, $options);
|
||||||
@@ -346,7 +346,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function text(string $name, $value = null, array $options = null): string
|
public function text(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
$options = $this->expandOptionArray($name, $label, $options);
|
$options = $this->expandOptionArray($name, $label, $options);
|
||||||
@@ -370,7 +370,7 @@ class ExpandedForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function textarea(string $name, $value = null, array $options = null): string
|
public function textarea(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
$options = $this->expandOptionArray($name, $label, $options);
|
$options = $this->expandOptionArray($name, $label, $options);
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ class AccountForm
|
|||||||
/**
|
/**
|
||||||
* Grouped dropdown list of all accounts that are valid as the destination of a withdrawal.
|
* Grouped dropdown list of all accounts that are valid as the destination of a withdrawal.
|
||||||
*/
|
*/
|
||||||
public function activeDepositDestinations(string $name, mixed $value = null, array $options = null): string
|
public function activeDepositDestinations(string $name, mixed $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$types = [AccountType::MORTGAGE, AccountType::DEBT, AccountType::CREDITCARD, AccountType::LOAN, AccountType::REVENUE];
|
$types = [AccountType::MORTGAGE, AccountType::DEBT, AccountType::CREDITCARD, AccountType::LOAN, AccountType::REVENUE];
|
||||||
$repository = $this->getAccountRepository();
|
$repository = $this->getAccountRepository();
|
||||||
@@ -55,7 +55,7 @@ class AccountForm
|
|||||||
return $this->select($name, $grouped, $value, $options);
|
return $this->select($name, $grouped, $value, $options);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function getAccountsGrouped(array $types, AccountRepositoryInterface $repository = null): array
|
private function getAccountsGrouped(array $types, ?AccountRepositoryInterface $repository = null): array
|
||||||
{
|
{
|
||||||
if (null === $repository) {
|
if (null === $repository) {
|
||||||
$repository = $this->getAccountRepository();
|
$repository = $this->getAccountRepository();
|
||||||
@@ -89,7 +89,7 @@ class AccountForm
|
|||||||
/**
|
/**
|
||||||
* Grouped dropdown list of all accounts that are valid as the destination of a withdrawal.
|
* Grouped dropdown list of all accounts that are valid as the destination of a withdrawal.
|
||||||
*/
|
*/
|
||||||
public function activeWithdrawalDestinations(string $name, mixed $value = null, array $options = null): string
|
public function activeWithdrawalDestinations(string $name, mixed $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$types = [AccountType::MORTGAGE, AccountType::DEBT, AccountType::CREDITCARD, AccountType::LOAN, AccountType::EXPENSE];
|
$types = [AccountType::MORTGAGE, AccountType::DEBT, AccountType::CREDITCARD, AccountType::LOAN, AccountType::EXPENSE];
|
||||||
$repository = $this->getAccountRepository();
|
$repository = $this->getAccountRepository();
|
||||||
@@ -107,7 +107,7 @@ class AccountForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function assetAccountCheckList(string $name, array $options = null): string
|
public function assetAccountCheckList(string $name, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
@@ -138,7 +138,7 @@ class AccountForm
|
|||||||
*
|
*
|
||||||
* @param mixed $value
|
* @param mixed $value
|
||||||
*/
|
*/
|
||||||
public function assetAccountList(string $name, $value = null, array $options = null): string
|
public function assetAccountList(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$types = [AccountType::ASSET, AccountType::DEFAULT];
|
$types = [AccountType::ASSET, AccountType::DEFAULT];
|
||||||
$grouped = $this->getAccountsGrouped($types);
|
$grouped = $this->getAccountsGrouped($types);
|
||||||
@@ -151,7 +151,7 @@ class AccountForm
|
|||||||
*
|
*
|
||||||
* @param mixed $value
|
* @param mixed $value
|
||||||
*/
|
*/
|
||||||
public function longAccountList(string $name, $value = null, array $options = null): string
|
public function longAccountList(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$types = [AccountType::ASSET, AccountType::DEFAULT, AccountType::MORTGAGE, AccountType::DEBT, AccountType::CREDITCARD, AccountType::LOAN];
|
$types = [AccountType::ASSET, AccountType::DEFAULT, AccountType::MORTGAGE, AccountType::DEBT, AccountType::CREDITCARD, AccountType::LOAN];
|
||||||
$grouped = $this->getAccountsGrouped($types);
|
$grouped = $this->getAccountsGrouped($types);
|
||||||
|
|||||||
@@ -42,7 +42,7 @@ class CurrencyForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function amount(string $name, $value = null, array $options = null): string
|
public function amount(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
return $this->currencyField($name, 'amount', $value, $options);
|
return $this->currencyField($name, 'amount', $value, $options);
|
||||||
}
|
}
|
||||||
@@ -50,7 +50,7 @@ class CurrencyForm
|
|||||||
/**
|
/**
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
protected function currencyField(string $name, string $view, mixed $value = null, array $options = null): string
|
protected function currencyField(string $name, string $view, mixed $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
$options = $this->expandOptionArray($name, $label, $options);
|
$options = $this->expandOptionArray($name, $label, $options);
|
||||||
@@ -106,7 +106,7 @@ class CurrencyForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function balanceAll(string $name, $value = null, array $options = null): string
|
public function balanceAll(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
return $this->allCurrencyField($name, 'balance', $value, $options);
|
return $this->allCurrencyField($name, 'balance', $value, $options);
|
||||||
}
|
}
|
||||||
@@ -118,7 +118,7 @@ class CurrencyForm
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
protected function allCurrencyField(string $name, string $view, $value = null, array $options = null): string
|
protected function allCurrencyField(string $name, string $view, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
$options = $this->expandOptionArray($name, $label, $options);
|
$options = $this->expandOptionArray($name, $label, $options);
|
||||||
@@ -173,7 +173,7 @@ class CurrencyForm
|
|||||||
*
|
*
|
||||||
* @param mixed $value
|
* @param mixed $value
|
||||||
*/
|
*/
|
||||||
public function currencyList(string $name, $value = null, array $options = null): string
|
public function currencyList(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
/** @var CurrencyRepositoryInterface $currencyRepos */
|
/** @var CurrencyRepositoryInterface $currencyRepos */
|
||||||
$currencyRepos = app(CurrencyRepositoryInterface::class);
|
$currencyRepos = app(CurrencyRepositoryInterface::class);
|
||||||
@@ -195,7 +195,7 @@ class CurrencyForm
|
|||||||
*
|
*
|
||||||
* @param mixed $value
|
* @param mixed $value
|
||||||
*/
|
*/
|
||||||
public function currencyListEmpty(string $name, $value = null, array $options = null): string
|
public function currencyListEmpty(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
/** @var CurrencyRepositoryInterface $currencyRepos */
|
/** @var CurrencyRepositoryInterface $currencyRepos */
|
||||||
$currencyRepos = app(CurrencyRepositoryInterface::class);
|
$currencyRepos = app(CurrencyRepositoryInterface::class);
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ trait FormSupport
|
|||||||
/**
|
/**
|
||||||
* @param mixed $selected
|
* @param mixed $selected
|
||||||
*/
|
*/
|
||||||
public function select(string $name, array $list = null, $selected = null, array $options = null): string
|
public function select(string $name, ?array $list = null, $selected = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$list ??= [];
|
$list ??= [];
|
||||||
$label = $this->label($name, $options);
|
$label = $this->label($name, $options);
|
||||||
@@ -55,7 +55,7 @@ trait FormSupport
|
|||||||
return $html;
|
return $html;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function label(string $name, array $options = null): string
|
protected function label(string $name, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
if (array_key_exists('label', $options)) {
|
if (array_key_exists('label', $options)) {
|
||||||
@@ -69,7 +69,7 @@ trait FormSupport
|
|||||||
/**
|
/**
|
||||||
* @param mixed $label
|
* @param mixed $label
|
||||||
*/
|
*/
|
||||||
protected function expandOptionArray(string $name, $label, array $options = null): array
|
protected function expandOptionArray(string $name, $label, ?array $options = null): array
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
$name = str_replace('[]', '', $name);
|
$name = str_replace('[]', '', $name);
|
||||||
|
|||||||
@@ -40,7 +40,7 @@ class PiggyBankForm
|
|||||||
*
|
*
|
||||||
* @param mixed $value
|
* @param mixed $value
|
||||||
*/
|
*/
|
||||||
public function piggyBankList(string $name, $value = null, array $options = null): string
|
public function piggyBankList(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
// make repositories
|
// make repositories
|
||||||
/** @var PiggyBankRepositoryInterface $repository */
|
/** @var PiggyBankRepositoryInterface $repository */
|
||||||
|
|||||||
@@ -34,7 +34,7 @@ class RuleForm
|
|||||||
{
|
{
|
||||||
use FormSupport;
|
use FormSupport;
|
||||||
|
|
||||||
public function ruleGroupList(string $name, mixed $value = null, array $options = null): string
|
public function ruleGroupList(string $name, mixed $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
/** @var RuleGroupRepositoryInterface $groupRepos */
|
/** @var RuleGroupRepositoryInterface $groupRepos */
|
||||||
$groupRepos = app(RuleGroupRepositoryInterface::class);
|
$groupRepos = app(RuleGroupRepositoryInterface::class);
|
||||||
@@ -54,7 +54,7 @@ class RuleForm
|
|||||||
/**
|
/**
|
||||||
* @param null $value
|
* @param null $value
|
||||||
*/
|
*/
|
||||||
public function ruleGroupListWithEmpty(string $name, $value = null, array $options = null): string
|
public function ruleGroupListWithEmpty(string $name, $value = null, ?array $options = null): string
|
||||||
{
|
{
|
||||||
$options ??= [];
|
$options ??= [];
|
||||||
$options['class'] = 'form-control';
|
$options['class'] = 'form-control';
|
||||||
|
|||||||
@@ -39,14 +39,14 @@ trait DateCalculation
|
|||||||
*/
|
*/
|
||||||
public function activeDaysLeft(Carbon $start, Carbon $end): int
|
public function activeDaysLeft(Carbon $start, Carbon $end): int
|
||||||
{
|
{
|
||||||
$difference = $start->diffInDays($end) + 1;
|
$difference = (int)($start->diffInDays($end, true) + 1);
|
||||||
$today = today(config('app.timezone'))->startOfDay();
|
$today = today(config('app.timezone'))->startOfDay();
|
||||||
|
|
||||||
if ($start->lte($today) && $end->gte($today)) {
|
if ($start->lte($today) && $end->gte($today)) {
|
||||||
$difference = $today->diffInDays($end);
|
$difference = $today->diffInDays($end);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0 === $difference ? 1 : $difference;
|
return (int) (0 === $difference ? 1 : $difference);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -56,20 +56,20 @@ trait DateCalculation
|
|||||||
*/
|
*/
|
||||||
protected function activeDaysPassed(Carbon $start, Carbon $end): int
|
protected function activeDaysPassed(Carbon $start, Carbon $end): int
|
||||||
{
|
{
|
||||||
$difference = $start->diffInDays($end) + 1;
|
$difference = $start->diffInDays($end, true) + 1;
|
||||||
$today = today(config('app.timezone'))->startOfDay();
|
$today = today(config('app.timezone'))->startOfDay();
|
||||||
|
|
||||||
if ($start->lte($today) && $end->gte($today)) {
|
if ($start->lte($today) && $end->gte($today)) {
|
||||||
$difference = $start->diffInDays($today) + 1;
|
$difference = $start->diffInDays($today, true) + 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $difference;
|
return (int) $difference;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function calculateStep(Carbon $start, Carbon $end): string
|
protected function calculateStep(Carbon $start, Carbon $end): string
|
||||||
{
|
{
|
||||||
$step = '1D';
|
$step = '1D';
|
||||||
$months = $start->diffInMonths($end);
|
$months = $start->diffInMonths($end, true);
|
||||||
if ($months > 3) {
|
if ($months > 3) {
|
||||||
$step = '1W';
|
$step = '1W';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,6 +25,7 @@ namespace FireflyIII\Support\Http\Controllers;
|
|||||||
|
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use FireflyIII\Exceptions\FireflyException;
|
use FireflyIII\Exceptions\FireflyException;
|
||||||
|
use Illuminate\Support\Facades\Log;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Trait GetConfigurationData
|
* Trait GetConfigurationData
|
||||||
@@ -82,6 +83,8 @@ trait GetConfigurationData
|
|||||||
{
|
{
|
||||||
$viewRange = app('navigation')->getViewRange(false);
|
$viewRange = app('navigation')->getViewRange(false);
|
||||||
|
|
||||||
|
Log::debug(sprintf('dateRange: the view range is "%s"', $viewRange));
|
||||||
|
|
||||||
/** @var Carbon $start */
|
/** @var Carbon $start */
|
||||||
$start = session('start');
|
$start = session('start');
|
||||||
|
|
||||||
@@ -97,6 +100,7 @@ trait GetConfigurationData
|
|||||||
// first range is the current range:
|
// first range is the current range:
|
||||||
$title => [$start, $end],
|
$title => [$start, $end],
|
||||||
];
|
];
|
||||||
|
Log::debug(sprintf('dateRange: the date range in the session is"%s" - "%s"', $start->format('Y-m-d'), $end->format('Y-m-d')));
|
||||||
|
|
||||||
// when current range is a custom range, add the current period as the next range.
|
// when current range is a custom range, add the current period as the next range.
|
||||||
if ($isCustom) {
|
if ($isCustom) {
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ class Navigation
|
|||||||
{
|
{
|
||||||
private Calculator $calculator;
|
private Calculator $calculator;
|
||||||
|
|
||||||
public function __construct(Calculator $calculator = null)
|
public function __construct(?Calculator $calculator = null)
|
||||||
{
|
{
|
||||||
$this->calculator = $calculator instanceof Calculator ? $calculator : new Calculator();
|
$this->calculator = $calculator instanceof Calculator ? $calculator : new Calculator();
|
||||||
}
|
}
|
||||||
@@ -161,7 +161,7 @@ class Navigation
|
|||||||
public function startOfPeriod(Carbon $theDate, string $repeatFreq): Carbon
|
public function startOfPeriod(Carbon $theDate, string $repeatFreq): Carbon
|
||||||
{
|
{
|
||||||
$date = clone $theDate;
|
$date = clone $theDate;
|
||||||
|
Log::debug(sprintf('Now in startOfPeriod("%s", "%s")', $date->toIso8601String(), $repeatFreq));
|
||||||
$functionMap = [
|
$functionMap = [
|
||||||
'1D' => 'startOfDay',
|
'1D' => 'startOfDay',
|
||||||
'daily' => 'startOfDay',
|
'daily' => 'startOfDay',
|
||||||
@@ -178,15 +178,32 @@ class Navigation
|
|||||||
'yearly' => 'startOfYear',
|
'yearly' => 'startOfYear',
|
||||||
'1Y' => 'startOfYear',
|
'1Y' => 'startOfYear',
|
||||||
];
|
];
|
||||||
|
|
||||||
|
$parameterMap = [
|
||||||
|
'startOfWeek' => [Carbon::MONDAY],
|
||||||
|
];
|
||||||
|
|
||||||
if (array_key_exists($repeatFreq, $functionMap)) {
|
if (array_key_exists($repeatFreq, $functionMap)) {
|
||||||
$function = $functionMap[$repeatFreq];
|
$function = $functionMap[$repeatFreq];
|
||||||
|
Log::debug(sprintf('Function is ->%s()', $function));
|
||||||
|
if (array_key_exists($function, $parameterMap)) {
|
||||||
|
Log::debug(sprintf('Parameter map, function becomes ->%s(%s)', $function, implode(', ', $parameterMap[$function])));
|
||||||
|
$date->{$function}($parameterMap[$function][0]);
|
||||||
|
Log::debug(sprintf('Result is "%s"', $date->toIso8601String()));
|
||||||
|
|
||||||
|
return $date;
|
||||||
|
}
|
||||||
|
|
||||||
$date->{$function}(); // @phpstan-ignore-line
|
$date->{$function}(); // @phpstan-ignore-line
|
||||||
|
Log::debug(sprintf('Result is "%s"', $date->toIso8601String()));
|
||||||
|
|
||||||
return $date;
|
return $date;
|
||||||
}
|
}
|
||||||
if ('half-year' === $repeatFreq || '6M' === $repeatFreq) {
|
if ('half-year' === $repeatFreq || '6M' === $repeatFreq) {
|
||||||
$skipTo = $date->month > 7 ? 6 : 0;
|
$skipTo = $date->month > 7 ? 6 : 0;
|
||||||
$date->startOfYear()->addMonths($skipTo);
|
$date->startOfYear()->addMonths($skipTo);
|
||||||
|
Log::debug(sprintf('Custom call for "%s": addMonths(%d)', $repeatFreq, $skipTo));
|
||||||
|
Log::debug(sprintf('Result is "%s"', $date->toIso8601String()));
|
||||||
|
|
||||||
return $date;
|
return $date;
|
||||||
}
|
}
|
||||||
@@ -202,10 +219,14 @@ class Navigation
|
|||||||
default => null,
|
default => null,
|
||||||
};
|
};
|
||||||
if (null !== $result) {
|
if (null !== $result) {
|
||||||
|
Log::debug(sprintf('Result is "%s"', $date->toIso8601String()));
|
||||||
|
|
||||||
return $result;
|
return $result;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ('custom' === $repeatFreq) {
|
if ('custom' === $repeatFreq) {
|
||||||
|
Log::debug(sprintf('Custom, result is "%s"', $date->toIso8601String()));
|
||||||
|
|
||||||
return $date; // the date is already at the start.
|
return $date; // the date is already at the start.
|
||||||
}
|
}
|
||||||
Log::error(sprintf('Cannot do startOfPeriod for $repeat_freq "%s"', $repeatFreq));
|
Log::error(sprintf('Cannot do startOfPeriod for $repeat_freq "%s"', $repeatFreq));
|
||||||
@@ -216,6 +237,7 @@ class Navigation
|
|||||||
public function endOfPeriod(Carbon $end, string $repeatFreq): Carbon
|
public function endOfPeriod(Carbon $end, string $repeatFreq): Carbon
|
||||||
{
|
{
|
||||||
$currentEnd = clone $end;
|
$currentEnd = clone $end;
|
||||||
|
Log::debug(sprintf('Now in endOfPeriod("%s", "%s").', $currentEnd->toIso8601String(), $repeatFreq));
|
||||||
|
|
||||||
$functionMap = [
|
$functionMap = [
|
||||||
'1D' => 'endOfDay',
|
'1D' => 'endOfDay',
|
||||||
@@ -252,7 +274,7 @@ class Navigation
|
|||||||
|
|
||||||
/** @var Carbon $tEnd */
|
/** @var Carbon $tEnd */
|
||||||
$tEnd = session('end', today(config('app.timezone'))->endOfMonth());
|
$tEnd = session('end', today(config('app.timezone'))->endOfMonth());
|
||||||
$diffInDays = $tStart->diffInDays($tEnd);
|
$diffInDays = (int) $tStart->diffInDays($tEnd, true);
|
||||||
}
|
}
|
||||||
Log::debug(sprintf('Diff in days is %d', $diffInDays));
|
Log::debug(sprintf('Diff in days is %d', $diffInDays));
|
||||||
$currentEnd->addDays($diffInDays);
|
$currentEnd->addDays($diffInDays);
|
||||||
@@ -296,6 +318,7 @@ class Navigation
|
|||||||
if (in_array($repeatFreq, $subDay, true)) {
|
if (in_array($repeatFreq, $subDay, true)) {
|
||||||
$currentEnd->subDay();
|
$currentEnd->subDay();
|
||||||
}
|
}
|
||||||
|
Log::debug(sprintf('Final result: %s', $currentEnd->toIso8601String()));
|
||||||
|
|
||||||
return $currentEnd;
|
return $currentEnd;
|
||||||
}
|
}
|
||||||
@@ -304,7 +327,7 @@ class Navigation
|
|||||||
{
|
{
|
||||||
$endOfMonth = $date->copy()->endOfMonth();
|
$endOfMonth = $date->copy()->endOfMonth();
|
||||||
|
|
||||||
return $date->diffInDays($endOfMonth);
|
return (int) $date->diffInDays($endOfMonth, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function diffInPeriods(string $period, int $skip, Carbon $beginning, Carbon $end): int
|
public function diffInPeriods(string $period, int $skip, Carbon $beginning, Carbon $end): int
|
||||||
@@ -317,12 +340,12 @@ class Navigation
|
|||||||
$end->format('Y-m-d')
|
$end->format('Y-m-d')
|
||||||
));
|
));
|
||||||
$map = [
|
$map = [
|
||||||
'daily' => 'floatDiffInDays',
|
'daily' => 'diffInDays',
|
||||||
'weekly' => 'floatDiffInWeeks',
|
'weekly' => 'diffInWeeks',
|
||||||
'monthly' => 'floatDiffInMonths',
|
'monthly' => 'diffInMonths',
|
||||||
'quarterly' => 'floatDiffInMonths',
|
'quarterly' => 'diffInMonths',
|
||||||
'half-year' => 'floatDiffInMonths',
|
'half-year' => 'diffInMonths',
|
||||||
'yearly' => 'floatDiffInYears',
|
'yearly' => 'diffInYears',
|
||||||
];
|
];
|
||||||
if (!array_key_exists($period, $map)) {
|
if (!array_key_exists($period, $map)) {
|
||||||
Log::warning(sprintf('No diffInPeriods for period "%s"', $period));
|
Log::warning(sprintf('No diffInPeriods for period "%s"', $period));
|
||||||
@@ -331,7 +354,7 @@ class Navigation
|
|||||||
}
|
}
|
||||||
$func = $map[$period];
|
$func = $map[$period];
|
||||||
// first do the diff
|
// first do the diff
|
||||||
$floatDiff = $beginning->{$func}($end); // @phpstan-ignore-line
|
$floatDiff = $beginning->{$func}($end, true); // @phpstan-ignore-line
|
||||||
|
|
||||||
// then correct for quarterly or half-year
|
// then correct for quarterly or half-year
|
||||||
if ('quarterly' === $period) {
|
if ('quarterly' === $period) {
|
||||||
@@ -442,13 +465,13 @@ class Navigation
|
|||||||
$format = $this->preferredCarbonFormat($start, $end);
|
$format = $this->preferredCarbonFormat($start, $end);
|
||||||
$displayFormat = (string)trans('config.month_and_day_js', [], $locale);
|
$displayFormat = (string)trans('config.month_and_day_js', [], $locale);
|
||||||
// increment by month (for year)
|
// increment by month (for year)
|
||||||
if ($start->diffInMonths($end) > 1) {
|
if ($start->diffInMonths($end, true) > 1) {
|
||||||
$increment = 'addMonth';
|
$increment = 'addMonth';
|
||||||
$displayFormat = (string)trans('config.month_js');
|
$displayFormat = (string)trans('config.month_js');
|
||||||
}
|
}
|
||||||
|
|
||||||
// increment by year (for multi year)
|
// increment by year (for multi-year)
|
||||||
if ($start->diffInMonths($end) > 12) {
|
if ($start->diffInMonths($end, true) > 12) {
|
||||||
$increment = 'addYear';
|
$increment = 'addYear';
|
||||||
$displayFormat = (string)trans('config.year_js');
|
$displayFormat = (string)trans('config.year_js');
|
||||||
}
|
}
|
||||||
@@ -471,11 +494,11 @@ class Navigation
|
|||||||
public function preferredCarbonFormat(Carbon $start, Carbon $end): string
|
public function preferredCarbonFormat(Carbon $start, Carbon $end): string
|
||||||
{
|
{
|
||||||
$format = 'Y-m-d';
|
$format = 'Y-m-d';
|
||||||
if ($start->diffInMonths($end) > 1) {
|
if ((int)$start->diffInMonths($end, true) > 1) {
|
||||||
$format = 'Y-m';
|
$format = 'Y-m';
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($start->diffInMonths($end) > 12) {
|
if ((int)$start->diffInMonths($end, true) > 12) {
|
||||||
$format = 'Y';
|
$format = 'Y';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -540,11 +563,11 @@ class Navigation
|
|||||||
{
|
{
|
||||||
$locale = app('steam')->getLocale();
|
$locale = app('steam')->getLocale();
|
||||||
$format = (string)trans('config.month_and_day_js', [], $locale);
|
$format = (string)trans('config.month_and_day_js', [], $locale);
|
||||||
if ($start->diffInMonths($end) > 1) {
|
if ($start->diffInMonths($end, true) > 1) {
|
||||||
$format = (string)trans('config.month_js', [], $locale);
|
$format = (string)trans('config.month_js', [], $locale);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($start->diffInMonths($end) > 12) {
|
if ($start->diffInMonths($end, true) > 12) {
|
||||||
$format = (string)trans('config.year_js', [], $locale);
|
$format = (string)trans('config.year_js', [], $locale);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -558,11 +581,11 @@ class Navigation
|
|||||||
public function preferredEndOfPeriod(Carbon $start, Carbon $end): string
|
public function preferredEndOfPeriod(Carbon $start, Carbon $end): string
|
||||||
{
|
{
|
||||||
$format = 'endOfDay';
|
$format = 'endOfDay';
|
||||||
if ($start->diffInMonths($end) > 1) {
|
if ((int)$start->diffInMonths($end, true) > 1) {
|
||||||
$format = 'endOfMonth';
|
$format = 'endOfMonth';
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($start->diffInMonths($end) > 12) {
|
if ((int)$start->diffInMonths($end, true) > 12) {
|
||||||
$format = 'endOfYear';
|
$format = 'endOfYear';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -576,11 +599,11 @@ class Navigation
|
|||||||
public function preferredRangeFormat(Carbon $start, Carbon $end): string
|
public function preferredRangeFormat(Carbon $start, Carbon $end): string
|
||||||
{
|
{
|
||||||
$format = '1D';
|
$format = '1D';
|
||||||
if ($start->diffInMonths($end) > 1) {
|
if ((int)$start->diffInMonths($end, true) > 1) {
|
||||||
$format = '1M';
|
$format = '1M';
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($start->diffInMonths($end) > 12) {
|
if ((int)$start->diffInMonths($end, true) > 12) {
|
||||||
$format = '1Y';
|
$format = '1Y';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -594,11 +617,11 @@ class Navigation
|
|||||||
public function preferredSqlFormat(Carbon $start, Carbon $end): string
|
public function preferredSqlFormat(Carbon $start, Carbon $end): string
|
||||||
{
|
{
|
||||||
$format = '%Y-%m-%d';
|
$format = '%Y-%m-%d';
|
||||||
if ($start->diffInMonths($end) > 1) {
|
if ((int)$start->diffInMonths($end, true) > 1) {
|
||||||
$format = '%Y-%m';
|
$format = '%Y-%m';
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($start->diffInMonths($end) > 12) {
|
if ((int)$start->diffInMonths($end, true) > 12) {
|
||||||
$format = '%Y';
|
$format = '%Y';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -608,7 +631,7 @@ class Navigation
|
|||||||
/**
|
/**
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
public function subtractPeriod(Carbon $theDate, string $repeatFreq, int $subtract = null): Carbon
|
public function subtractPeriod(Carbon $theDate, string $repeatFreq, ?int $subtract = null): Carbon
|
||||||
{
|
{
|
||||||
$subtract ??= 1;
|
$subtract ??= 1;
|
||||||
$date = clone $theDate;
|
$date = clone $theDate;
|
||||||
@@ -654,7 +677,7 @@ class Navigation
|
|||||||
|
|
||||||
/** @var Carbon $tEnd */
|
/** @var Carbon $tEnd */
|
||||||
$tEnd = session('end', today(config('app.timezone'))->endOfMonth());
|
$tEnd = session('end', today(config('app.timezone'))->endOfMonth());
|
||||||
$diffInDays = $tStart->diffInDays($tEnd);
|
$diffInDays = (int) $tStart->diffInDays($tEnd, true);
|
||||||
$date->subDays($diffInDays * $subtract);
|
$date->subDays($diffInDays * $subtract);
|
||||||
|
|
||||||
return $date;
|
return $date;
|
||||||
|
|||||||
@@ -31,6 +31,6 @@ class UrlValidator
|
|||||||
{
|
{
|
||||||
public static function isValidWebhookURL(string $url): bool
|
public static function isValidWebhookURL(string $url): bool
|
||||||
{
|
{
|
||||||
return str_starts_with($url, 'https://hooks.slack.com/services/') || str_starts_with($url, 'https://discord.com/api/webhooks/');
|
return str_starts_with($url, 'https://hooks.slack.com/services/') || str_starts_with($url, 'https://discord.com/api/webhooks/') || str_ends_with($url, '/slack');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -126,8 +126,8 @@ class ParseDateString
|
|||||||
default => $today,
|
default => $today,
|
||||||
'yesterday' => $today->subDay(),
|
'yesterday' => $today->subDay(),
|
||||||
'tomorrow' => $today->addDay(),
|
'tomorrow' => $today->addDay(),
|
||||||
'start of this week' => $today->startOfWeek(),
|
'start of this week' => $today->startOfWeek(Carbon::MONDAY),
|
||||||
'end of this week' => $today->endOfWeek(),
|
'end of this week' => $today->endOfWeek(Carbon::SUNDAY),
|
||||||
'start of this month' => $today->startOfMonth(),
|
'start of this month' => $today->startOfMonth(),
|
||||||
'end of this month' => $today->endOfMonth(),
|
'end of this month' => $today->endOfMonth(),
|
||||||
'start of this quarter' => $today->startOfQuarter(),
|
'start of this quarter' => $today->startOfQuarter(),
|
||||||
|
|||||||
52
app/Support/Request/GetSortInstructions.php
Normal file
52
app/Support/Request/GetSortInstructions.php
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
<?php
|
||||||
|
/*
|
||||||
|
* GetSortInstructions.php
|
||||||
|
* Copyright (c) 2024 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/.
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
namespace FireflyIII\Support\Request;
|
||||||
|
|
||||||
|
trait GetSortInstructions
|
||||||
|
{
|
||||||
|
final public function getSortInstructions(string $key): array
|
||||||
|
{
|
||||||
|
$allowed = config(sprintf('firefly.sorting.allowed.%s', $key));
|
||||||
|
$set = $this->get('sorting', []);
|
||||||
|
$result = [];
|
||||||
|
if (0 === count($set)) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
foreach ($set as $info) {
|
||||||
|
$column = $info['column'] ?? 'NOPE';
|
||||||
|
$direction = $info['direction'] ?? 'NOPE';
|
||||||
|
if ('asc' !== $direction && 'desc' !== $direction) {
|
||||||
|
// skip invalid direction
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (false === in_array($column, $allowed, true)) {
|
||||||
|
// skip invalid column
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$result[$column] = $direction;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -177,7 +177,7 @@ class OperatorQuerySearch implements SearchInterface
|
|||||||
default:
|
default:
|
||||||
app('log')->error(sprintf('Cannot handle node %s', $class));
|
app('log')->error(sprintf('Cannot handle node %s', $class));
|
||||||
|
|
||||||
throw new FireflyException(sprintf('Firefly III search cant handle "%s"-nodes', $class));
|
throw new FireflyException(sprintf('Firefly III search can\'t handle "%s"-nodes', $class));
|
||||||
|
|
||||||
case Subquery::class:
|
case Subquery::class:
|
||||||
// loop all notes in subquery:
|
// loop all notes in subquery:
|
||||||
|
|||||||
@@ -851,7 +851,7 @@ class Steam
|
|||||||
return number_format((float)$value, 0, '.', '');
|
return number_format((float)$value, 0, '.', '');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function opposite(string $amount = null): ?string
|
public function opposite(?string $amount = null): ?string
|
||||||
{
|
{
|
||||||
if (null === $amount) {
|
if (null === $amount) {
|
||||||
return null;
|
return null;
|
||||||
|
|||||||
@@ -88,7 +88,7 @@ class OAuthKeys
|
|||||||
public static function generateKeys(): void
|
public static function generateKeys(): void
|
||||||
{
|
{
|
||||||
\Artisan::registerCommand(new KeysCommand());
|
\Artisan::registerCommand(new KeysCommand());
|
||||||
\Artisan::call('passport:keys');
|
\Artisan::call('firefly-iii:laravel-passport-keys');
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function storeKeysInDB(): void
|
public static function storeKeysInDB(): void
|
||||||
|
|||||||
@@ -87,7 +87,7 @@ class AmountFormat extends AbstractExtension
|
|||||||
{
|
{
|
||||||
return new TwigFunction(
|
return new TwigFunction(
|
||||||
'formatAmountByAccount',
|
'formatAmountByAccount',
|
||||||
static function (AccountModel $account, string $amount, bool $coloured = null): string {
|
static function (AccountModel $account, string $amount, ?bool $coloured = null): string {
|
||||||
$coloured ??= true;
|
$coloured ??= true;
|
||||||
|
|
||||||
/** @var AccountRepositoryInterface $accountRepos */
|
/** @var AccountRepositoryInterface $accountRepos */
|
||||||
@@ -107,7 +107,7 @@ class AmountFormat extends AbstractExtension
|
|||||||
{
|
{
|
||||||
return new TwigFunction(
|
return new TwigFunction(
|
||||||
'formatAmountBySymbol',
|
'formatAmountBySymbol',
|
||||||
static function (string $amount, string $symbol, int $decimalPlaces = null, bool $coloured = null): string {
|
static function (string $amount, string $symbol, ?int $decimalPlaces = null, ?bool $coloured = null): string {
|
||||||
$decimalPlaces ??= 2;
|
$decimalPlaces ??= 2;
|
||||||
$coloured ??= true;
|
$coloured ??= true;
|
||||||
$currency = new TransactionCurrency();
|
$currency = new TransactionCurrency();
|
||||||
@@ -127,7 +127,7 @@ class AmountFormat extends AbstractExtension
|
|||||||
{
|
{
|
||||||
return new TwigFunction(
|
return new TwigFunction(
|
||||||
'formatAmountByCurrency',
|
'formatAmountByCurrency',
|
||||||
static function (TransactionCurrency $currency, string $amount, bool $coloured = null): string {
|
static function (TransactionCurrency $currency, string $amount, ?bool $coloured = null): string {
|
||||||
$coloured ??= true;
|
$coloured ??= true;
|
||||||
|
|
||||||
return app('amount')->formatAnything($currency, $amount, $coloured);
|
return app('amount')->formatAnything($currency, $amount, $coloured);
|
||||||
|
|||||||
@@ -54,11 +54,12 @@ class AddTag implements ActionInterface
|
|||||||
/** @var User $user */
|
/** @var User $user */
|
||||||
$user = User::find($journal['user_id']);
|
$user = User::find($journal['user_id']);
|
||||||
$factory->setUser($user);
|
$factory->setUser($user);
|
||||||
$tag = $factory->findOrCreate($this->action->action_value);
|
$tagName = $this->action->getValue($journal);
|
||||||
|
$tag = $factory->findOrCreate($tagName);
|
||||||
|
|
||||||
if (null === $tag) {
|
if (null === $tag) {
|
||||||
// could not find, could not create tag.
|
// could not find, could not create tag.
|
||||||
event(new RuleActionFailedOnArray($this->action, $journal, trans('rules.find_or_create_tag_failed', ['tag' => $this->action->action_value])));
|
event(new RuleActionFailedOnArray($this->action, $journal, trans('rules.find_or_create_tag_failed', ['tag' => $tagName])));
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -84,7 +85,7 @@ class AddTag implements ActionInterface
|
|||||||
app('log')->debug(
|
app('log')->debug(
|
||||||
sprintf('RuleAction AddTag fired but tag %d ("%s") was already added to journal %d.', $tag->id, $tag->tag, $journal['transaction_journal_id'])
|
sprintf('RuleAction AddTag fired but tag %d ("%s") was already added to journal %d.', $tag->id, $tag->tag, $journal['transaction_journal_id'])
|
||||||
);
|
);
|
||||||
event(new RuleActionFailedOnArray($this->action, $journal, trans('rules.tag_already_added', ['tag' => $this->action->action_value])));
|
event(new RuleActionFailedOnArray($this->action, $journal, trans('rules.tag_already_added', ['tag' => $tagName])));
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,12 +26,16 @@ namespace FireflyIII\TransactionRules\Actions;
|
|||||||
use FireflyIII\Events\TriggeredAuditLog;
|
use FireflyIII\Events\TriggeredAuditLog;
|
||||||
use FireflyIII\Models\RuleAction;
|
use FireflyIII\Models\RuleAction;
|
||||||
use FireflyIII\Models\TransactionJournal;
|
use FireflyIII\Models\TransactionJournal;
|
||||||
|
use FireflyIII\TransactionRules\Traits\RefreshNotesTrait;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class AppendDescription.
|
* Class AppendDescription.
|
||||||
|
* TODO Can be replaced (and migrated) to action "set description" with a prefilled expression
|
||||||
*/
|
*/
|
||||||
class AppendDescription implements ActionInterface
|
class AppendDescription implements ActionInterface
|
||||||
{
|
{
|
||||||
|
use RefreshNotesTrait;
|
||||||
|
|
||||||
private RuleAction $action;
|
private RuleAction $action;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -44,7 +48,9 @@ class AppendDescription implements ActionInterface
|
|||||||
|
|
||||||
public function actOnArray(array $journal): bool
|
public function actOnArray(array $journal): bool
|
||||||
{
|
{
|
||||||
$description = sprintf('%s %s', $journal['description'], $this->action->action_value);
|
$this->refreshNotes($journal);
|
||||||
|
$append = $this->action->getValue($journal);
|
||||||
|
$description = sprintf('%s %s', $journal['description'], $append);
|
||||||
\DB::table('transaction_journals')->where('id', $journal['transaction_journal_id'])->limit(1)->update(['description' => $description]);
|
\DB::table('transaction_journals')->where('id', $journal['transaction_journal_id'])->limit(1)->update(['description' => $description]);
|
||||||
|
|
||||||
// event for audit log entry
|
// event for audit log entry
|
||||||
|
|||||||
@@ -29,12 +29,16 @@ use FireflyIII\Events\TriggeredAuditLog;
|
|||||||
use FireflyIII\Models\Note;
|
use FireflyIII\Models\Note;
|
||||||
use FireflyIII\Models\RuleAction;
|
use FireflyIII\Models\RuleAction;
|
||||||
use FireflyIII\Models\TransactionJournal;
|
use FireflyIII\Models\TransactionJournal;
|
||||||
|
use FireflyIII\TransactionRules\Traits\RefreshNotesTrait;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class AppendDescriptionToNotes
|
* Class AppendDescriptionToNotes
|
||||||
|
* TODO Can be replaced (and migrated) to action "set notes" with a prefilled expression
|
||||||
*/
|
*/
|
||||||
class AppendDescriptionToNotes implements ActionInterface
|
class AppendDescriptionToNotes implements ActionInterface
|
||||||
{
|
{
|
||||||
|
use RefreshNotesTrait;
|
||||||
|
|
||||||
private RuleAction $action;
|
private RuleAction $action;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -47,6 +51,8 @@ class AppendDescriptionToNotes implements ActionInterface
|
|||||||
|
|
||||||
public function actOnArray(array $journal): bool
|
public function actOnArray(array $journal): bool
|
||||||
{
|
{
|
||||||
|
$this->refreshNotes($journal);
|
||||||
|
|
||||||
/** @var null|TransactionJournal $object */
|
/** @var null|TransactionJournal $object */
|
||||||
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
||||||
if (null === $object) {
|
if (null === $object) {
|
||||||
|
|||||||
@@ -27,12 +27,16 @@ use FireflyIII\Events\TriggeredAuditLog;
|
|||||||
use FireflyIII\Models\Note;
|
use FireflyIII\Models\Note;
|
||||||
use FireflyIII\Models\RuleAction;
|
use FireflyIII\Models\RuleAction;
|
||||||
use FireflyIII\Models\TransactionJournal;
|
use FireflyIII\Models\TransactionJournal;
|
||||||
|
use FireflyIII\TransactionRules\Traits\RefreshNotesTrait;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class AppendNotes.
|
* Class AppendNotes.
|
||||||
|
* TODO Can be replaced (and migrated) to action "set notes" with a prefilled expression
|
||||||
*/
|
*/
|
||||||
class AppendNotes implements ActionInterface
|
class AppendNotes implements ActionInterface
|
||||||
{
|
{
|
||||||
|
use RefreshNotesTrait;
|
||||||
|
|
||||||
private RuleAction $action;
|
private RuleAction $action;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -45,6 +49,7 @@ class AppendNotes implements ActionInterface
|
|||||||
|
|
||||||
public function actOnArray(array $journal): bool
|
public function actOnArray(array $journal): bool
|
||||||
{
|
{
|
||||||
|
$this->refreshNotes($journal);
|
||||||
$dbNote = Note::where('noteable_id', (int)$journal['transaction_journal_id'])
|
$dbNote = Note::where('noteable_id', (int)$journal['transaction_journal_id'])
|
||||||
->where('noteable_type', TransactionJournal::class)
|
->where('noteable_type', TransactionJournal::class)
|
||||||
->first(['notes.*'])
|
->first(['notes.*'])
|
||||||
@@ -55,15 +60,16 @@ class AppendNotes implements ActionInterface
|
|||||||
$dbNote->noteable_type = TransactionJournal::class;
|
$dbNote->noteable_type = TransactionJournal::class;
|
||||||
$dbNote->text = '';
|
$dbNote->text = '';
|
||||||
}
|
}
|
||||||
app('log')->debug(sprintf('RuleAction AppendNotes appended "%s" to "%s".', $this->action->action_value, $dbNote->text));
|
|
||||||
$before = $dbNote->text;
|
$before = $dbNote->text;
|
||||||
$text = sprintf('%s%s', $dbNote->text, $this->action->action_value);
|
$append = $this->action->getValue($journal);
|
||||||
|
$text = sprintf('%s%s', $dbNote->text, $append);
|
||||||
$dbNote->text = $text;
|
$dbNote->text = $text;
|
||||||
$dbNote->save();
|
$dbNote->save();
|
||||||
|
|
||||||
/** @var TransactionJournal $object */
|
/** @var TransactionJournal $object */
|
||||||
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
||||||
|
|
||||||
|
app('log')->debug(sprintf('RuleAction AppendNotes appended "%s" to "%s".', $append, $before));
|
||||||
event(new TriggeredAuditLog($this->action->rule, $object, 'update_notes', $before, $text));
|
event(new TriggeredAuditLog($this->action->rule, $object, 'update_notes', $before, $text));
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|||||||
@@ -30,13 +30,16 @@ use FireflyIII\Models\Note;
|
|||||||
use FireflyIII\Models\RuleAction;
|
use FireflyIII\Models\RuleAction;
|
||||||
use FireflyIII\Models\TransactionJournal;
|
use FireflyIII\Models\TransactionJournal;
|
||||||
use FireflyIII\Support\Request\ConvertsDataTypes;
|
use FireflyIII\Support\Request\ConvertsDataTypes;
|
||||||
|
use FireflyIII\TransactionRules\Traits\RefreshNotesTrait;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class AppendNotesToDescription
|
* Class AppendNotesToDescription
|
||||||
|
* TODO Can be replaced (and migrated) to action "set description" with a prefilled expression
|
||||||
*/
|
*/
|
||||||
class AppendNotesToDescription implements ActionInterface
|
class AppendNotesToDescription implements ActionInterface
|
||||||
{
|
{
|
||||||
use ConvertsDataTypes;
|
use ConvertsDataTypes;
|
||||||
|
use RefreshNotesTrait;
|
||||||
|
|
||||||
private RuleAction $action;
|
private RuleAction $action;
|
||||||
|
|
||||||
@@ -51,6 +54,7 @@ class AppendNotesToDescription implements ActionInterface
|
|||||||
public function actOnArray(array $journal): bool
|
public function actOnArray(array $journal): bool
|
||||||
{
|
{
|
||||||
app('log')->debug('Now in AppendNotesToDescription');
|
app('log')->debug('Now in AppendNotesToDescription');
|
||||||
|
$this->refreshNotes($journal);
|
||||||
|
|
||||||
/** @var null|TransactionJournal $object */
|
/** @var null|TransactionJournal $object */
|
||||||
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
||||||
|
|||||||
@@ -52,6 +52,8 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
|
|
||||||
public function actOnArray(array $journal): bool
|
public function actOnArray(array $journal): bool
|
||||||
{
|
{
|
||||||
|
$actionValue = $this->action->getValue($journal);
|
||||||
|
|
||||||
// make object from array (so the data is fresh).
|
// make object from array (so the data is fresh).
|
||||||
/** @var null|TransactionJournal $object */
|
/** @var null|TransactionJournal $object */
|
||||||
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
||||||
@@ -82,7 +84,7 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
app('log')->debug('Going to transform a withdrawal to a deposit.');
|
app('log')->debug('Going to transform a withdrawal to a deposit.');
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$res = $this->convertWithdrawalArray($object);
|
$res = $this->convertWithdrawalArray($object, $actionValue);
|
||||||
} catch (FireflyException $e) {
|
} catch (FireflyException $e) {
|
||||||
app('log')->debug('Could not convert withdrawal to deposit.');
|
app('log')->debug('Could not convert withdrawal to deposit.');
|
||||||
app('log')->error($e->getMessage());
|
app('log')->error($e->getMessage());
|
||||||
@@ -99,7 +101,7 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
app('log')->debug('Going to transform a transfer to a deposit.');
|
app('log')->debug('Going to transform a transfer to a deposit.');
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$res = $this->convertTransferArray($object);
|
$res = $this->convertTransferArray($object, $actionValue);
|
||||||
} catch (FireflyException $e) {
|
} catch (FireflyException $e) {
|
||||||
app('log')->debug('Could not convert transfer to deposit.');
|
app('log')->debug('Could not convert transfer to deposit.');
|
||||||
app('log')->error($e->getMessage());
|
app('log')->error($e->getMessage());
|
||||||
@@ -122,7 +124,7 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
private function convertWithdrawalArray(TransactionJournal $journal): bool
|
private function convertWithdrawalArray(TransactionJournal $journal, string $actionValue = ''): bool
|
||||||
{
|
{
|
||||||
$user = $journal->user;
|
$user = $journal->user;
|
||||||
|
|
||||||
@@ -139,7 +141,7 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
|
|
||||||
// get the action value, or use the original destination name in case the action value is empty:
|
// get the action value, or use the original destination name in case the action value is empty:
|
||||||
// this becomes a new or existing (revenue) account, which is the source of the new deposit.
|
// this becomes a new or existing (revenue) account, which is the source of the new deposit.
|
||||||
$opposingName = '' === $this->action->action_value ? $destAccount->name : $this->action->action_value;
|
$opposingName = '' === $actionValue ? $destAccount->name : $actionValue;
|
||||||
// we check all possible source account types if one exists:
|
// we check all possible source account types if one exists:
|
||||||
$validTypes = config('firefly.expected_source_types.source.Deposit');
|
$validTypes = config('firefly.expected_source_types.source.Deposit');
|
||||||
$opposingAccount = $repository->findByName($opposingName, $validTypes);
|
$opposingAccount = $repository->findByName($opposingName, $validTypes);
|
||||||
@@ -147,7 +149,7 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
$opposingAccount = $factory->findOrCreate($opposingName, AccountType::REVENUE);
|
$opposingAccount = $factory->findOrCreate($opposingName, AccountType::REVENUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
app('log')->debug(sprintf('ConvertToDeposit. Action value is "%s", new opposing name is "%s"', $this->action->action_value, $opposingAccount->name));
|
app('log')->debug(sprintf('ConvertToDeposit. Action value is "%s", new opposing name is "%s"', $actionValue, $opposingAccount->name));
|
||||||
|
|
||||||
// update the source transaction and put in the new revenue ID.
|
// update the source transaction and put in the new revenue ID.
|
||||||
\DB::table('transactions')
|
\DB::table('transactions')
|
||||||
@@ -211,7 +213,7 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
private function convertTransferArray(TransactionJournal $journal): bool
|
private function convertTransferArray(TransactionJournal $journal, string $actionValue = ''): bool
|
||||||
{
|
{
|
||||||
$user = $journal->user;
|
$user = $journal->user;
|
||||||
|
|
||||||
@@ -227,7 +229,7 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
|
|
||||||
// get the action value, or use the original source name in case the action value is empty:
|
// get the action value, or use the original source name in case the action value is empty:
|
||||||
// this becomes a new or existing (revenue) account, which is the source of the new deposit.
|
// this becomes a new or existing (revenue) account, which is the source of the new deposit.
|
||||||
$opposingName = '' === $this->action->action_value ? $sourceAccount->name : $this->action->action_value;
|
$opposingName = '' === $actionValue ? $sourceAccount->name : $actionValue;
|
||||||
// we check all possible source account types if one exists:
|
// we check all possible source account types if one exists:
|
||||||
$validTypes = config('firefly.expected_source_types.source.Deposit');
|
$validTypes = config('firefly.expected_source_types.source.Deposit');
|
||||||
$opposingAccount = $repository->findByName($opposingName, $validTypes);
|
$opposingAccount = $repository->findByName($opposingName, $validTypes);
|
||||||
@@ -235,7 +237,7 @@ class ConvertToDeposit implements ActionInterface
|
|||||||
$opposingAccount = $factory->findOrCreate($opposingName, AccountType::REVENUE);
|
$opposingAccount = $factory->findOrCreate($opposingName, AccountType::REVENUE);
|
||||||
}
|
}
|
||||||
|
|
||||||
app('log')->debug(sprintf('ConvertToDeposit. Action value is "%s", revenue name is "%s"', $this->action->action_value, $opposingAccount->name));
|
app('log')->debug(sprintf('ConvertToDeposit. Action value is "%s", revenue name is "%s"', $actionValue, $opposingAccount->name));
|
||||||
|
|
||||||
// update source transaction(s) to be revenue account
|
// update source transaction(s) to be revenue account
|
||||||
\DB::table('transactions')
|
\DB::table('transactions')
|
||||||
|
|||||||
@@ -55,6 +55,8 @@ class ConvertToTransfer implements ActionInterface
|
|||||||
*/
|
*/
|
||||||
public function actOnArray(array $journal): bool
|
public function actOnArray(array $journal): bool
|
||||||
{
|
{
|
||||||
|
$accountName = $this->action->getValue($journal);
|
||||||
|
|
||||||
// make object from array (so the data is fresh).
|
// make object from array (so the data is fresh).
|
||||||
/** @var null|TransactionJournal $object */
|
/** @var null|TransactionJournal $object */
|
||||||
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
||||||
@@ -102,7 +104,7 @@ class ConvertToTransfer implements ActionInterface
|
|||||||
$expectedType = $this->getDestinationType($journalId);
|
$expectedType = $this->getDestinationType($journalId);
|
||||||
// Deposit? Replace source with account with same type as destination.
|
// Deposit? Replace source with account with same type as destination.
|
||||||
}
|
}
|
||||||
$opposing = $repository->findByName($this->action->action_value, [$expectedType]);
|
$opposing = $repository->findByName($accountName, [$expectedType]);
|
||||||
|
|
||||||
if (null === $opposing) {
|
if (null === $opposing) {
|
||||||
app('log')->error(
|
app('log')->error(
|
||||||
@@ -110,11 +112,11 @@ class ConvertToTransfer implements ActionInterface
|
|||||||
'Journal #%d cannot be converted because no valid %s account with name "%s" exists (rule #%d).',
|
'Journal #%d cannot be converted because no valid %s account with name "%s" exists (rule #%d).',
|
||||||
$expectedType,
|
$expectedType,
|
||||||
$journalId,
|
$journalId,
|
||||||
$this->action->action_value,
|
$accountName,
|
||||||
$this->action->rule_id
|
$this->action->rule_id
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
event(new RuleActionFailedOnArray($this->action, $journal, trans('rules.no_valid_opposing', ['name' => $this->action->action_value])));
|
event(new RuleActionFailedOnArray($this->action, $journal, trans('rules.no_valid_opposing', ['name' => $accountName])));
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -52,6 +52,8 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
|
|
||||||
public function actOnArray(array $journal): bool
|
public function actOnArray(array $journal): bool
|
||||||
{
|
{
|
||||||
|
$actionValue = $this->action->getValue($journal);
|
||||||
|
|
||||||
// make object from array (so the data is fresh).
|
// make object from array (so the data is fresh).
|
||||||
/** @var null|TransactionJournal $object */
|
/** @var null|TransactionJournal $object */
|
||||||
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
$object = TransactionJournal::where('user_id', $journal['user_id'])->find($journal['transaction_journal_id']);
|
||||||
@@ -85,7 +87,7 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
app('log')->debug('Going to transform a deposit to a withdrawal.');
|
app('log')->debug('Going to transform a deposit to a withdrawal.');
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$res = $this->convertDepositArray($object);
|
$res = $this->convertDepositArray($object, $actionValue);
|
||||||
} catch (FireflyException $e) {
|
} catch (FireflyException $e) {
|
||||||
app('log')->debug('Could not convert transfer to deposit.');
|
app('log')->debug('Could not convert transfer to deposit.');
|
||||||
app('log')->error($e->getMessage());
|
app('log')->error($e->getMessage());
|
||||||
@@ -101,7 +103,7 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
app('log')->debug('Going to transform a transfer to a withdrawal.');
|
app('log')->debug('Going to transform a transfer to a withdrawal.');
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$res = $this->convertTransferArray($object);
|
$res = $this->convertTransferArray($object, $actionValue);
|
||||||
} catch (FireflyException $e) {
|
} catch (FireflyException $e) {
|
||||||
app('log')->debug('Could not convert transfer to deposit.');
|
app('log')->debug('Could not convert transfer to deposit.');
|
||||||
app('log')->error($e->getMessage());
|
app('log')->error($e->getMessage());
|
||||||
@@ -117,7 +119,7 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
/**
|
/**
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
private function convertDepositArray(TransactionJournal $journal): bool
|
private function convertDepositArray(TransactionJournal $journal, string $actionValue = ''): bool
|
||||||
{
|
{
|
||||||
$user = $journal->user;
|
$user = $journal->user;
|
||||||
|
|
||||||
@@ -133,7 +135,7 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
|
|
||||||
// get the action value, or use the original source name in case the action value is empty:
|
// get the action value, or use the original source name in case the action value is empty:
|
||||||
// this becomes a new or existing (expense) account, which is the destination of the new withdrawal.
|
// this becomes a new or existing (expense) account, which is the destination of the new withdrawal.
|
||||||
$opposingName = '' === $this->action->action_value ? $sourceAccount->name : $this->action->action_value;
|
$opposingName = '' === $actionValue ? $sourceAccount->name : $actionValue;
|
||||||
// we check all possible source account types if one exists:
|
// we check all possible source account types if one exists:
|
||||||
$validTypes = config('firefly.expected_source_types.destination.Withdrawal');
|
$validTypes = config('firefly.expected_source_types.destination.Withdrawal');
|
||||||
$opposingAccount = $repository->findByName($opposingName, $validTypes);
|
$opposingAccount = $repository->findByName($opposingName, $validTypes);
|
||||||
@@ -141,7 +143,7 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
$opposingAccount = $factory->findOrCreate($opposingName, AccountType::EXPENSE);
|
$opposingAccount = $factory->findOrCreate($opposingName, AccountType::EXPENSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
app('log')->debug(sprintf('ConvertToWithdrawal. Action value is "%s", expense name is "%s"', $this->action->action_value, $opposingName));
|
app('log')->debug(sprintf('ConvertToWithdrawal. Action value is "%s", expense name is "%s"', $actionValue, $opposingName));
|
||||||
|
|
||||||
// update source transaction(s) to be the original destination account
|
// update source transaction(s) to be the original destination account
|
||||||
\DB::table('transactions')
|
\DB::table('transactions')
|
||||||
@@ -203,7 +205,7 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
*
|
*
|
||||||
* @throws FireflyException
|
* @throws FireflyException
|
||||||
*/
|
*/
|
||||||
private function convertTransferArray(TransactionJournal $journal): bool
|
private function convertTransferArray(TransactionJournal $journal, string $actionValue = ''): bool
|
||||||
{
|
{
|
||||||
// find or create expense account.
|
// find or create expense account.
|
||||||
$user = $journal->user;
|
$user = $journal->user;
|
||||||
@@ -219,7 +221,7 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
|
|
||||||
// get the action value, or use the original source name in case the action value is empty:
|
// get the action value, or use the original source name in case the action value is empty:
|
||||||
// this becomes a new or existing (expense) account, which is the destination of the new withdrawal.
|
// this becomes a new or existing (expense) account, which is the destination of the new withdrawal.
|
||||||
$opposingName = '' === $this->action->action_value ? $destAccount->name : $this->action->action_value;
|
$opposingName = '' === $actionValue ? $destAccount->name : $actionValue;
|
||||||
// we check all possible source account types if one exists:
|
// we check all possible source account types if one exists:
|
||||||
$validTypes = config('firefly.expected_source_types.destination.Withdrawal');
|
$validTypes = config('firefly.expected_source_types.destination.Withdrawal');
|
||||||
$opposingAccount = $repository->findByName($opposingName, $validTypes);
|
$opposingAccount = $repository->findByName($opposingName, $validTypes);
|
||||||
@@ -227,7 +229,7 @@ class ConvertToWithdrawal implements ActionInterface
|
|||||||
$opposingAccount = $factory->findOrCreate($opposingName, AccountType::EXPENSE);
|
$opposingAccount = $factory->findOrCreate($opposingName, AccountType::EXPENSE);
|
||||||
}
|
}
|
||||||
|
|
||||||
app('log')->debug(sprintf('ConvertToWithdrawal. Action value is "%s", destination name is "%s"', $this->action->action_value, $opposingName));
|
app('log')->debug(sprintf('ConvertToWithdrawal. Action value is "%s", destination name is "%s"', $actionValue, $opposingName));
|
||||||
|
|
||||||
// update destination transaction(s) to be new expense account.
|
// update destination transaction(s) to be new expense account.
|
||||||
\DB::table('transactions')
|
\DB::table('transactions')
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ class LinkToBill implements ActionInterface
|
|||||||
/** @var BillRepositoryInterface $repository */
|
/** @var BillRepositoryInterface $repository */
|
||||||
$repository = app(BillRepositoryInterface::class);
|
$repository = app(BillRepositoryInterface::class);
|
||||||
$repository->setUser($user);
|
$repository->setUser($user);
|
||||||
$billName = (string)$this->action->action_value;
|
$billName = $this->action->getValue($journal);
|
||||||
$bill = $repository->findByName($billName);
|
$bill = $repository->findByName($billName);
|
||||||
|
|
||||||
if (null !== $bill && TransactionType::WITHDRAWAL === $journal['transaction_type_type']) {
|
if (null !== $bill && TransactionType::WITHDRAWAL === $journal['transaction_type_type']) {
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ use FireflyIII\Models\TransactionJournal;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Class MoveDescriptionToNotes
|
* Class MoveDescriptionToNotes
|
||||||
|
* TODO Can be replaced (and migrated) to action "set notes" with a prefilled expression
|
||||||
*/
|
*/
|
||||||
class MoveDescriptionToNotes implements ActionInterface
|
class MoveDescriptionToNotes implements ActionInterface
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -36,6 +36,7 @@ use FireflyIII\Support\Request\ConvertsDataTypes;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Class MoveNotesToDescription
|
* Class MoveNotesToDescription
|
||||||
|
* TODO Can be replaced (and migrated) to action "set notes" with a prefilled expression
|
||||||
*/
|
*/
|
||||||
class MoveNotesToDescription implements ActionInterface
|
class MoveNotesToDescription implements ActionInterface
|
||||||
{
|
{
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user