mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-10-25 21:16:47 +00:00 
			
		
		
		
	Compare commits
	
		
			54 Commits
		
	
	
		
			develop-20
			...
			develop-20
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 4bd1aab86d | ||
|  | 60362cb60c | ||
|  | 27f740bf98 | ||
|  | 5e15747a5b | ||
|  | 8a6eaad2bb | ||
|  | 5ab52d9f66 | ||
|  | 21a327bf08 | ||
|  | 15dd175394 | ||
|  | 3f35305beb | ||
|  | 768d8b1515 | ||
|  | 1c19428a12 | ||
|  | c204533195 | ||
|  | 6d89485792 | ||
|  | 949d818bad | ||
|  | a12e200a0a | ||
|  | b4039b1f13 | ||
|  | 32921e15b1 | ||
|  | 4f7cc7d53b | ||
|  | 0986bfbc34 | ||
|  | 663202bfc6 | ||
|  | 6f63ddf5b0 | ||
|  | a9805b144a | ||
|  | e1b8b9b3ae | ||
|  | d57327fd11 | ||
|  | 15ac69bfad | ||
|  | a5bd28f8d4 | ||
|  | b2516ca1b4 | ||
|  | 053b46ae63 | ||
|  | 6e836aceec | ||
|  | 0e8bcd2e79 | ||
|  | bd1f8b2497 | ||
|  | 19dfcf7139 | ||
|  | ef7a3287bb | ||
|  | 2900049498 | ||
|  | 04d1e8fd59 | ||
|  | 9d2f57e40a | ||
|  | ae366341cc | ||
|  | 3766128cb8 | ||
|  | 950c60d55c | ||
|  | 4b2807de48 | ||
|  | 649736cb31 | ||
|  | 6a121a8a78 | ||
|  | f69b9ac9da | ||
|  | 23d70a2fac | ||
|  | d178ff9de0 | ||
|  | 3ecad3457f | ||
|  | fa6c621968 | ||
|  | df863b6cff | ||
|  | 9316ff3e51 | ||
|  | bfd91f8ee6 | ||
|  | 13e5d25cfe | ||
|  | b13030420b | ||
|  | 44d1e8181c | ||
|  | 63b34c1853 | 
							
								
								
									
										12
									
								
								.ci/php-cs-fixer/composer.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										12
									
								
								.ci/php-cs-fixer/composer.lock
									
									
									
										generated
									
									
									
								
							| @@ -406,16 +406,16 @@ | ||||
|         }, | ||||
|         { | ||||
|             "name": "friendsofphp/php-cs-fixer", | ||||
|             "version": "v3.67.0", | ||||
|             "version": "v3.68.1", | ||||
|             "source": { | ||||
|                 "type": "git", | ||||
|                 "url": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer.git", | ||||
|                 "reference": "0ad34c75d1172f7d30320460e803887981830cbf" | ||||
|                 "reference": "b9db2b2ea3cdba7201067acee46f984ef2397cff" | ||||
|             }, | ||||
|             "dist": { | ||||
|                 "type": "zip", | ||||
|                 "url": "https://api.github.com/repos/PHP-CS-Fixer/PHP-CS-Fixer/zipball/0ad34c75d1172f7d30320460e803887981830cbf", | ||||
|                 "reference": "0ad34c75d1172f7d30320460e803887981830cbf", | ||||
|                 "url": "https://api.github.com/repos/PHP-CS-Fixer/PHP-CS-Fixer/zipball/b9db2b2ea3cdba7201067acee46f984ef2397cff", | ||||
|                 "reference": "b9db2b2ea3cdba7201067acee46f984ef2397cff", | ||||
|                 "shasum": "" | ||||
|             }, | ||||
|             "require": { | ||||
| @@ -497,7 +497,7 @@ | ||||
|             ], | ||||
|             "support": { | ||||
|                 "issues": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/issues", | ||||
|                 "source": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/tree/v3.67.0" | ||||
|                 "source": "https://github.com/PHP-CS-Fixer/PHP-CS-Fixer/tree/v3.68.1" | ||||
|             }, | ||||
|             "funding": [ | ||||
|                 { | ||||
| @@ -505,7 +505,7 @@ | ||||
|                     "type": "github" | ||||
|                 } | ||||
|             ], | ||||
|             "time": "2025-01-08T10:17:40+00:00" | ||||
|             "time": "2025-01-17T09:20:36+00:00" | ||||
|         }, | ||||
|         { | ||||
|             "name": "psr/container", | ||||
|   | ||||
| @@ -16,6 +16,9 @@ parameters: | ||||
|     - '#Dynamic call to static method#' # all the Laravel ORM things depend on this. | ||||
|     - identifier: varTag.nativeType | ||||
|     - identifier: varTag.type | ||||
|     - | ||||
|         identifier: larastan.noEnvCallsOutsideOfConfig | ||||
|         path: ../app/Console/Commands/System/CreatesDatabase.php | ||||
|     - identifier: missingType.iterableValue # not interesting enough to fix. | ||||
|     - identifier: missingType.generics # not interesting enough to fix. | ||||
|     - "#Parameter \\#[1-2] \\$num[1-2] of function bc[a-z]+ expects numeric-string, [a-z\\-|&]+ given#" | ||||
| @@ -28,49 +31,6 @@ parameters: | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::withTrashed#' | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\HasMany::accountTypeIn#' | ||||
|     - '#Call to an undefined method Illuminate\\Database\\Eloquent\\Relations\\BelongsTo::withTrashed#' | ||||
|     # - '#Control structures using switch should not be used.#' # switch is fine in some cases. | ||||
|     # - '#with no value type specified in iterable type array#' # remove this rule when all other issues are solved. | ||||
|     # - '#has no value type specified in iterable type array#' # remove this rule when all other issues are solved. | ||||
|     # - '#is not allowed to extend#' | ||||
|     # - '#does not specify its types#' | ||||
|     # - '#switch is forbidden to use#' | ||||
|     # - '#is neither abstract nor final#' | ||||
|     # - '#on left side of \?\?\= always exists and is not nullable#' | ||||
|     # - '#has a nullable return type declaration#' # perhaps throw errors instead? | ||||
|     # - '#with a nullable type declaration#' # decide what action should be if param is null. | ||||
|     # - '#with null as default value#' | ||||
|     # - | ||||
|     #     message: '#Constructor in [a-zA-Z0-9\\_]+ has parameter \$[a-zA-Z0-9\\_]+ with default value#' | ||||
|     #     paths: | ||||
|     #         - ../app/Exceptions/IntervalException.php | ||||
|     #         - ../app/Support/Navigation.php | ||||
|     # - | ||||
|     #     message: '#but containers should not be injected#' | ||||
|     #     paths: | ||||
|     #         - ../app/Support/Authentication/RemoteUserGuard.php | ||||
|     # - | ||||
|     #     message: '#Function compact\(\) should not be used#' # too useful in template rendering. | ||||
|     #     paths: | ||||
|     #         - ../app/Generator/Report/Account/MonthReportGenerator.php | ||||
|     #         - ../app/Generator/Report/Audit/MonthReportGenerator.php | ||||
|     #         - ../app/Generator/Report/Budget/MonthReportGenerator.php | ||||
|     #         - ../app/Generator/Report/Category/MonthReportGenerator.php | ||||
|     #         - ../app/Generator/Report/Standard/MonthReportGenerator.php | ||||
|     #         - ../app/Generator/Report/Standard/MultiYearReportGenerator.php | ||||
|     #         - ../app/Generator/Report/Standard/YearReportGenerator.php | ||||
|     #         - ../app/Generator/Report/Tag/MonthReportGenerator.php | ||||
|     #         - ../app/Http/Controllers/Account/*.php | ||||
|     #         - ../app/Http/Controllers/Admin/*.php | ||||
|     #         - ../app/Http/Controllers/*.php | ||||
|     #         - ../app/Support/ExpandedForm.php | ||||
|     #         - ../app/Support/Form/AccountForm.php | ||||
|     #         - ../app/Support/Form/CurrencyForm.php | ||||
|     #         - ../app/Support/Form/FormSupport.php | ||||
|     # - | ||||
|     #     message: '#Either catch a more specific exception#' | ||||
|     #     paths: | ||||
|     #         - ../app/Support/Form/FormSupport.php | ||||
|  | ||||
|  | ||||
|   # The level 8 is the highest level. original was 5 | ||||
|   # 7 is more than enough, higher just leaves NULL things. | ||||
|   | ||||
| @@ -41,6 +41,7 @@ use Illuminate\Http\JsonResponse; | ||||
| class AccountController extends Controller | ||||
| { | ||||
|     use AccountFilter; | ||||
|     protected array $accepts = ['application/json']; | ||||
| 
 | ||||
|     /** @var array<int, string> */ | ||||
|     private array                      $balanceTypes; | ||||
| @@ -84,12 +85,12 @@ class AccountController extends Controller | ||||
|         /** @var Account $account */ | ||||
|         foreach ($result as $account) { | ||||
|             $nameWithBalance = $account->name; | ||||
|             $currency        = $this->repository->getAccountCurrency($account) ?? $this->defaultCurrency; | ||||
|             $currency        = $this->repository->getAccountCurrency($account) ?? $this->nativeCurrency; | ||||
|             $useCurrency     = $currency; | ||||
|             if (in_array($account->accountType->type, $this->balanceTypes, true)) { | ||||
|                 $balance         = Steam::finalAccountBalance($account, $date); | ||||
|                 $key             = $this->convertToNative && $currency->id !== $this->defaultCurrency->id ? 'native_balance' : 'balance'; | ||||
|                 $useCurrency     = $this->convertToNative && $currency->id !== $this->defaultCurrency->id ? $this->defaultCurrency : $currency; | ||||
|                 $key             = $this->convertToNative && $currency->id !== $this->nativeCurrency->id ? 'native_balance' : 'balance'; | ||||
|                 $useCurrency     = $this->convertToNative && $currency->id !== $this->nativeCurrency->id ? $this->nativeCurrency : $currency; | ||||
|                 $amount          = $balance[$key] ?? '0'; | ||||
|                 $nameWithBalance = sprintf( | ||||
|                     '%s (%s)', | ||||
| @@ -123,6 +124,6 @@ class AccountController extends Controller | ||||
|             } | ||||
|         ); | ||||
| 
 | ||||
|         return response()->json($return); | ||||
|         return response()->api($return); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -74,6 +74,6 @@ class BillController extends Controller | ||||
|             } | ||||
|         ); | ||||
| 
 | ||||
|         return response()->json($filtered->toArray()); | ||||
|         return response()->api($filtered->toArray()); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -73,6 +73,6 @@ class BudgetController extends Controller | ||||
|             } | ||||
|         ); | ||||
| 
 | ||||
|         return response()->json($filtered); | ||||
|         return response()->api($filtered->toArray()); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -73,6 +73,6 @@ class CategoryController extends Controller | ||||
|             } | ||||
|         ); | ||||
| 
 | ||||
|         return response()->json($filtered); | ||||
|         return response()->api($filtered->toArray()); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -77,7 +77,7 @@ class CurrencyController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($result); | ||||
|         return response()->api($result); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -103,6 +103,6 @@ class CurrencyController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($result); | ||||
|         return response()->api($result); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -75,6 +75,6 @@ class ObjectGroupController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($return); | ||||
|         return response()->api($return); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -87,7 +87,7 @@ class PiggyBankController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($response); | ||||
|         return response()->api($response); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -124,6 +124,6 @@ class PiggyBankController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($response); | ||||
|         return response()->api($response); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -73,6 +73,6 @@ class RecurrenceController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($response); | ||||
|         return response()->api($response); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -72,6 +72,6 @@ class RuleController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($response); | ||||
|         return response()->api($response); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -72,6 +72,6 @@ class RuleGroupController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($response); | ||||
|         return response()->api($response); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -75,6 +75,6 @@ class TagController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($array); | ||||
|         return response()->api($array); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -84,7 +84,7 @@ class TransactionController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($array); | ||||
|         return response()->api($array); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -122,6 +122,6 @@ class TransactionController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($array); | ||||
|         return response()->api($array); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -72,6 +72,6 @@ class TransactionTypeController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($array); | ||||
|         return response()->api($array); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -97,8 +97,8 @@ class AccountController extends Controller | ||||
| 
 | ||||
|         /** @var Account $account */ | ||||
|         foreach ($accounts as $account) { | ||||
|             $currency     = $this->repository->getAccountCurrency($account) ?? $this->defaultCurrency; | ||||
|             $field        = $this->convertToNative && $currency->id !== $this->defaultCurrency->id ? 'native_balance' : 'balance'; | ||||
|             $currency     = $this->repository->getAccountCurrency($account) ?? $this->nativeCurrency; | ||||
|             $field        = $this->convertToNative && $currency->id !== $this->nativeCurrency->id ? 'native_balance' : 'balance'; | ||||
|             $currentSet   = [ | ||||
|                 'label'                   => $account->name, | ||||
|                 'currency_id'             => (string) $currency->id, | ||||
|   | ||||
| @@ -26,16 +26,24 @@ namespace FireflyIII\Api\V1\Controllers; | ||||
| 
 | ||||
| use Carbon\Carbon; | ||||
| use Carbon\Exceptions\InvalidFormatException; | ||||
| use FireflyIII\Exceptions\BadHttpHeaderException; | ||||
| use FireflyIII\Models\Preference; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use FireflyIII\Transformers\V2\AbstractTransformer; | ||||
| use FireflyIII\User; | ||||
| use Illuminate\Database\Eloquent\Model; | ||||
| use Illuminate\Foundation\Auth\Access\AuthorizesRequests; | ||||
| use Illuminate\Foundation\Bus\DispatchesJobs; | ||||
| use Illuminate\Foundation\Validation\ValidatesRequests; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use Illuminate\Routing\Controller as BaseController; | ||||
| use Illuminate\Support\Collection; | ||||
| use League\Fractal\Manager; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| use League\Fractal\Resource\Collection as FractalCollection; | ||||
| use League\Fractal\Resource\Item; | ||||
| use League\Fractal\Serializer\JsonApiSerializer; | ||||
| use Symfony\Component\HttpFoundation\Exception\BadRequestException; | ||||
| use Symfony\Component\HttpFoundation\ParameterBag; | ||||
| @@ -52,13 +60,15 @@ abstract class Controller extends BaseController | ||||
|     use DispatchesJobs; | ||||
|     use ValidatesRequests; | ||||
| 
 | ||||
|     protected const string CONTENT_TYPE    = 'application/vnd.api+json'; | ||||
|     protected const string CONTENT_TYPE      = 'application/vnd.api+json'; | ||||
|     protected const string JSON_CONTENT_TYPE = 'application/json'; | ||||
| 
 | ||||
|     /** @var array<int, string> */ | ||||
|     protected array        $allowedSort; | ||||
|     protected ParameterBag $parameters; | ||||
|     protected bool        $convertToNative = false; | ||||
|     protected TransactionCurrency $defaultCurrency; | ||||
|     protected bool        $convertToNative   = false; | ||||
|     protected array $accepts                 = ['application/json']; | ||||
|     protected TransactionCurrency $nativeCurrency; | ||||
| 
 | ||||
|     /** | ||||
|      * Controller constructor. | ||||
| @@ -73,11 +83,17 @@ abstract class Controller extends BaseController | ||||
|                 if (auth()->check()) { | ||||
|                     $language              = Steam::getLanguage(); | ||||
|                     $this->convertToNative = Amount::convertToNative(); | ||||
|                     $this->defaultCurrency = Amount::getDefaultCurrency(); | ||||
|                     $this->nativeCurrency  = Amount::getNativeCurrency(); | ||||
|                     app()->setLocale($language); | ||||
| 
 | ||||
|                 } | ||||
| 
 | ||||
| 
 | ||||
|                 // filter down what this endpoint accepts.
 | ||||
|                 if (!$request->accepts($this->accepts)) { | ||||
|                     throw new BadHttpHeaderException(sprintf('Sorry, Accept header "%s" is not something this endpoint can provide.', $request->header('Accept'))); | ||||
|                 } | ||||
| 
 | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
| @@ -141,7 +157,15 @@ abstract class Controller extends BaseController | ||||
|                 $value = null; | ||||
|             } | ||||
|             if (null !== $value) { | ||||
|                 $bag->set($integer, (int) $value); | ||||
|                 $value = (int) $value; | ||||
|                 if ($value < 1) { | ||||
|                     $value = 1; | ||||
|                 } | ||||
|                 if ($value > 2 ** 16) { | ||||
|                     $value = 2 ** 16; | ||||
|                 } | ||||
| 
 | ||||
|                 $bag->set($integer, $value); | ||||
|             } | ||||
|             if (null === $value | ||||
|                 && 'limit' === $integer // @phpstan-ignore-line
 | ||||
| @@ -223,4 +247,45 @@ abstract class Controller extends BaseController | ||||
| 
 | ||||
|         return $manager; | ||||
|     } | ||||
| 
 | ||||
|     final protected function jsonApiList(string $key, LengthAwarePaginator $paginator, AbstractTransformer $transformer): array | ||||
|     { | ||||
|         $manager  = new Manager(); | ||||
|         $baseUrl  = sprintf('%s/api/v1/', request()->getSchemeAndHttpHost()); | ||||
| 
 | ||||
|         // TODO add stuff to path?
 | ||||
| 
 | ||||
|         $manager->setSerializer(new JsonApiSerializer($baseUrl)); | ||||
| 
 | ||||
|         $objects  = $paginator->getCollection(); | ||||
| 
 | ||||
|         // 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.
 | ||||
|         $objects  = $transformer->collectMetaData($objects); | ||||
|         $paginator->setCollection($objects); | ||||
| 
 | ||||
|         $resource = new FractalCollection($objects, $transformer, $key); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return $manager->createData($resource)->toArray(); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Returns a JSON API object and returns it. | ||||
|      * | ||||
|      * @param array<int, mixed>|Model $object | ||||
|      */ | ||||
|     final protected function jsonApiObject(string $key, array|Model $object, AbstractTransformer $transformer): array | ||||
|     { | ||||
|         // create some objects:
 | ||||
|         $manager  = new Manager(); | ||||
|         $baseUrl  = sprintf('%s/api/v1', request()->getSchemeAndHttpHost()); | ||||
|         $manager->setSerializer(new JsonApiSerializer($baseUrl)); | ||||
| 
 | ||||
|         $transformer->collectMetaData(new Collection([$object])); | ||||
| 
 | ||||
|         $resource = new Item($object, $transformer, $key); | ||||
| 
 | ||||
|         return $manager->createData($resource)->toArray(); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -70,7 +70,7 @@ class BillController extends Controller | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $response        = []; | ||||
| 
 | ||||
|         // get all bills:
 | ||||
| @@ -133,7 +133,7 @@ class BillController extends Controller | ||||
|         $start           = $request->getStart(); | ||||
|         $end             = $request->getEnd(); | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $response        = []; | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|   | ||||
| @@ -48,7 +48,7 @@ class PeriodController extends Controller | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type)
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|   | ||||
| @@ -69,7 +69,7 @@ class TagController extends Controller | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|   | ||||
| @@ -47,7 +47,7 @@ class PeriodController extends Controller | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type)
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|   | ||||
| @@ -69,7 +69,7 @@ class TagController extends Controller | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|   | ||||
| @@ -47,7 +47,7 @@ class PeriodController extends Controller | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type)
 | ||||
|         $collector       = app(GroupCollectorInterface::class); | ||||
|   | ||||
| @@ -67,7 +67,7 @@ class TagController extends Controller | ||||
|         $end             = $request->getEnd(); | ||||
|         $response        = []; | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
| 
 | ||||
| 
 | ||||
|         // collect all expenses in this period (regardless of type) by the given bills and accounts.
 | ||||
|   | ||||
| @@ -25,6 +25,7 @@ declare(strict_types=1); | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\Budget; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Enums\TransactionTypeEnum; | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Helpers\Collector\GroupCollectorInterface; | ||||
| use FireflyIII\Models\Budget; | ||||
| @@ -208,6 +209,8 @@ class ListController extends Controller | ||||
|         $collector    = app(GroupCollectorInterface::class); | ||||
|         $collector | ||||
|             ->setUser($admin) | ||||
|             // withdrawals only
 | ||||
|             ->setTypes([TransactionTypeEnum::WITHDRAWAL->value]) | ||||
|             // filter on budget.
 | ||||
|             ->withoutBudget() | ||||
|             // all info needed for the API:
 | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| 
 | ||||
| /* | ||||
|  * DestroyController.php | ||||
|  * Copyright (c) 2024 james@firefly-iii.org. | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -22,10 +22,13 @@ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Controllers\Model\ExchangeRate; | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Api\V2\Request\Model\ExchangeRate\DestroyRequest; | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate\DestroyRequest; | ||||
| use FireflyIII\Enums\UserRoleEnum; | ||||
| use FireflyIII\Exceptions\ValidationException; | ||||
| use FireflyIII\Models\CurrencyExchangeRate; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\UserGroups\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| @@ -36,6 +39,8 @@ class DestroyController extends Controller | ||||
| { | ||||
|     use ValidatesUserGroupTrait; | ||||
| 
 | ||||
|     protected array $acceptedRoles   = [UserRoleEnum::OWNER]; | ||||
| 
 | ||||
|     public const string RESOURCE_KEY = 'exchange-rates'; | ||||
| 
 | ||||
|     private ExchangeRateRepositoryInterface $repository; | ||||
| @@ -56,6 +61,9 @@ class DestroyController extends Controller | ||||
|     public function destroy(DestroyRequest $request, TransactionCurrency $from, TransactionCurrency $to): JsonResponse | ||||
|     { | ||||
|         $date = $request->getDate(); | ||||
|         if (null === $date) { | ||||
|             throw new ValidationException('Date is required'); | ||||
|         } | ||||
|         $rate = $this->repository->getSpecificRateOnDate($from, $to, $date); | ||||
|         if (null === $rate) { | ||||
|             throw new NotFoundHttpException(); | ||||
| @@ -64,4 +72,11 @@ class DestroyController extends Controller | ||||
| 
 | ||||
|         return response()->json([], 204); | ||||
|     } | ||||
| 
 | ||||
|     public function destroySingle(CurrencyExchangeRate $exchangeRate): JsonResponse | ||||
|     { | ||||
|         $this->repository->deleteRate($exchangeRate); | ||||
| 
 | ||||
|         return response()->json([], 204); | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * ShowController.php | ||||
|  * Copyright (c) 2023 james@firefly-iii.org | ||||
|  * IndexController.php | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -17,12 +17,12 @@ | ||||
|  * 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/>. | ||||
|  * along with this program.  If not, see https://www.gnu.org/licenses/. | ||||
|  */ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Controllers\Model\ExchangeRate; | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Repositories\UserGroups\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| @@ -38,7 +38,7 @@ class IndexController extends Controller | ||||
| { | ||||
|     use ValidatesUserGroupTrait; | ||||
| 
 | ||||
|     public const string RESOURCE_KEY = 'exchange-rates'; | ||||
|     public const string RESOURCE_KEY = 'currency_exchange_rates'; | ||||
| 
 | ||||
|     private ExchangeRateRepositoryInterface $repository; | ||||
| 
 | ||||
| @@ -62,9 +62,6 @@ class IndexController extends Controller | ||||
|         $count       = $piggies->count(); | ||||
|         $piggies     = $piggies->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
|         $paginator   = new LengthAwarePaginator($piggies, $count, $pageSize, $this->parameters->get('page')); | ||||
| 
 | ||||
|         var_dump('here we are'); | ||||
| 
 | ||||
|         $transformer = new ExchangeRateTransformer(); | ||||
|         $transformer->setParameters($this->parameters); // give params to transformer
 | ||||
| 
 | ||||
| @@ -2,7 +2,7 @@ | ||||
| 
 | ||||
| /* | ||||
|  * ShowController.php | ||||
|  * Copyright (c) 2023 james@firefly-iii.org | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -17,14 +17,15 @@ | ||||
|  * 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/>. | ||||
|  * along with this program.  If not, see https://www.gnu.org/licenses/. | ||||
|  */ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Controllers\Model\ExchangeRate; | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Models\CurrencyExchangeRate; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| use FireflyIII\Repositories\UserGroups\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| @@ -73,4 +74,15 @@ class ShowController extends Controller | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| 
 | ||||
|     public function showSingle(CurrencyExchangeRate $exchangeRate): JsonResponse | ||||
|     { | ||||
|         $transformer = new ExchangeRateTransformer(); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         return response() | ||||
|             ->api($this->jsonApiObject(self::RESOURCE_KEY, $exchangeRate, $transformer)) | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * DestroyController.php | ||||
|  * Copyright (c) 2024 james@firefly-iii.org. | ||||
|  * StoreController.php | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -22,10 +22,10 @@ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Controllers\Model\ExchangeRate; | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate\StoreRequest; | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Api\V2\Request\Model\ExchangeRate\StoreRequest; | ||||
| use FireflyIII\Repositories\UserGroups\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| use FireflyIII\Transformers\V2\ExchangeRateTransformer; | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * DestroyController.php | ||||
|  * Copyright (c) 2024 james@firefly-iii.org. | ||||
|  * UpdateController.php | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -22,10 +22,10 @@ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Controllers\Model\ExchangeRate; | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate\UpdateRequest; | ||||
| use FireflyIII\Api\V2\Controllers\Controller; | ||||
| use FireflyIII\Api\V2\Request\Model\ExchangeRate\UpdateRequest; | ||||
| use FireflyIII\Models\CurrencyExchangeRate; | ||||
| use FireflyIII\Repositories\UserGroups\ExchangeRate\ExchangeRateRepositoryInterface; | ||||
| use FireflyIII\Support\Http\Api\ValidatesUserGroupTrait; | ||||
| @@ -107,7 +107,7 @@ class ShowController extends Controller | ||||
|         /** @var User $user */ | ||||
|         $user        = auth()->user(); | ||||
|         $manager     = $this->getManager(); | ||||
|         $this->parameters->set('defaultCurrency', $this->defaultCurrency); | ||||
|         $this->parameters->set('nativeCurrency', $this->nativeCurrency); | ||||
| 
 | ||||
|         // update fields with user info.
 | ||||
|         $currency->refreshForUser($user); | ||||
| @@ -123,7 +123,7 @@ class ShowController extends Controller | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/currencies/getDefaultCurrency
 | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/currencies/getNativeCurrency
 | ||||
|      * | ||||
|      * Show a currency. | ||||
|      * | ||||
| @@ -134,7 +134,7 @@ class ShowController extends Controller | ||||
|         /** @var User $user */ | ||||
|         $user        = auth()->user(); | ||||
|         $manager     = $this->getManager(); | ||||
|         $currency    = $this->defaultCurrency; | ||||
|         $currency    = $this->nativeCurrency; | ||||
| 
 | ||||
|         // update fields with user info.
 | ||||
|         $currency->refreshForUser($user); | ||||
|   | ||||
| @@ -100,7 +100,7 @@ class UpdateController extends Controller | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/currencies/defaultCurrency
 | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/currencies/nativeCurrency
 | ||||
|      * | ||||
|      * Make the currency a default currency. | ||||
|      * | ||||
|   | ||||
							
								
								
									
										71
									
								
								app/Api/V1/Controllers/Models/UserGroup/IndexController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								app/Api/V1/Controllers/Models/UserGroup/IndexController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * IndexController.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\V1\Controllers\Models\UserGroup; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Data\DateRequest; | ||||
| use FireflyIII\Repositories\UserGroup\UserGroupRepositoryInterface; | ||||
| use FireflyIII\Transformers\UserGroupTransformer; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| 
 | ||||
| class IndexController extends Controller | ||||
| { | ||||
|     public const string RESOURCE_KEY = 'user_groups'; | ||||
| 
 | ||||
|     private UserGroupRepositoryInterface $repository; | ||||
| 
 | ||||
|     /** | ||||
|      * AccountController constructor. | ||||
|      */ | ||||
|     public function __construct() | ||||
|     { | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->repository = app(UserGroupRepositoryInterface::class); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     public function index(DateRequest $request): JsonResponse | ||||
|     { | ||||
|         $administrations = $this->repository->get(); | ||||
|         $pageSize        = $this->parameters->get('limit'); | ||||
|         $count           = $administrations->count(); | ||||
|         $administrations = $administrations->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
|         $paginator       = new LengthAwarePaginator($administrations, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $transformer     = new UserGroupTransformer(); | ||||
| 
 | ||||
|         $transformer->setParameters($this->parameters); // give params to transformer
 | ||||
| 
 | ||||
|         return response() | ||||
|             ->json($this->jsonApiList(self::RESOURCE_KEY, $paginator, $transformer)) | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										64
									
								
								app/Api/V1/Controllers/Models/UserGroup/ShowController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								app/Api/V1/Controllers/Models/UserGroup/ShowController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * ShowController.php | ||||
|  * Copyright (c) 2021 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\V1\Controllers\Models\UserGroup; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Models\UserGroup; | ||||
| use FireflyIII\Repositories\Webhook\WebhookRepositoryInterface; | ||||
| use FireflyIII\Transformers\UserGroupTransformer; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| /** | ||||
|  * Class ShowController | ||||
|  */ | ||||
| class ShowController extends Controller | ||||
| { | ||||
|     public const string RESOURCE_KEY = 'user_groups'; | ||||
|     private WebhookRepositoryInterface $repository; | ||||
| 
 | ||||
|     public function __construct() | ||||
|     { | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->repository = app(WebhookRepositoryInterface::class); | ||||
|                 $this->repository->setUser(auth()->user()); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     public function show(UserGroup $userGroup): JsonResponse | ||||
|     { | ||||
|         $transformer = new UserGroupTransformer(); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         return response() | ||||
|             ->api($this->jsonApiObject(self::RESOURCE_KEY, $userGroup, $transformer)) | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| } | ||||
							
								
								
									
										71
									
								
								app/Api/V1/Controllers/Models/UserGroup/UpdateController.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								app/Api/V1/Controllers/Models/UserGroup/UpdateController.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * UpdateController.php | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
|  * This program is free software: you can redistribute it and/or modify | ||||
|  * it under the terms of the GNU Affero General Public License as | ||||
|  * published by the Free Software Foundation, either version 3 of the | ||||
|  * License, or (at your option) any later version. | ||||
|  * | ||||
|  * This program is distributed in the hope that it will be useful, | ||||
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
|  * GNU Affero General Public License for more details. | ||||
|  * | ||||
|  * You should have received a copy of the GNU Affero General Public License | ||||
|  * along with this program.  If not, see https://www.gnu.org/licenses/. | ||||
|  */ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Controllers\Models\UserGroup; | ||||
| 
 | ||||
| use FireflyIII\Api\V1\Controllers\Controller; | ||||
| use FireflyIII\Api\V1\Requests\Models\UserGroup\UpdateRequest; | ||||
| use FireflyIII\Models\UserGroup; | ||||
| use FireflyIII\Repositories\UserGroup\UserGroupRepositoryInterface; | ||||
| use FireflyIII\Transformers\UserGroupTransformer; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| 
 | ||||
| class UpdateController extends Controller | ||||
| { | ||||
|     public const string RESOURCE_KEY = 'user_groups'; | ||||
| 
 | ||||
|     private UserGroupRepositoryInterface $repository; | ||||
| 
 | ||||
|     /** | ||||
|      * AccountController constructor. | ||||
|      */ | ||||
|     public function __construct() | ||||
|     { | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 $this->repository = app(UserGroupRepositoryInterface::class); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     public function update(UpdateRequest $request, UserGroup $userGroup): JsonResponse | ||||
|     { | ||||
|         app('log')->debug(sprintf('Now in %s', __METHOD__)); | ||||
|         $data        = $request->getData(); | ||||
|         $userGroup   = $this->repository->update($userGroup, $data); | ||||
|         $userGroup->refresh(); | ||||
|         app('preferences')->mark(); | ||||
| 
 | ||||
|         $transformer = new UserGroupTransformer(); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         return response() | ||||
|             ->api($this->jsonApiObject(self::RESOURCE_KEY, $userGroup, $transformer)) | ||||
|             ->header('Content-Type', self::CONTENT_TYPE) | ||||
|         ; | ||||
|     } | ||||
| } | ||||
| @@ -124,7 +124,7 @@ class BasicController extends Controller | ||||
|     { | ||||
|         // some config settings
 | ||||
|         $convertToNative = Amount::convertToNative(); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         // prep some arrays:
 | ||||
|         $incomes         = []; | ||||
|         $expenses        = []; | ||||
|   | ||||
| @@ -58,7 +58,7 @@ class AboutController extends Controller | ||||
|                            'driver'      => $currentDriver, | ||||
|                        ]; | ||||
| 
 | ||||
|         return response()->api(['data' => $data])->header('Content-Type', self::CONTENT_TYPE); | ||||
|         return response()->api(['data' => $data])->header('Content-Type', self::JSON_CONTENT_TYPE); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|   | ||||
| @@ -86,7 +86,7 @@ class ConfigurationController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json($return); | ||||
|         return response()->api($return); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -142,7 +142,7 @@ class ConfigurationController extends Controller | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
|         return response()->json(['data' => $data])->header('Content-Type', self::CONTENT_TYPE); | ||||
|         return response()->api(['data' => $data])->header('Content-Type', self::JSON_CONTENT_TYPE); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -173,6 +173,6 @@ class ConfigurationController extends Controller | ||||
|             'editable' => true, | ||||
|         ]; | ||||
| 
 | ||||
|         return response()->json(['data' => $data])->header('Content-Type', self::CONTENT_TYPE); | ||||
|         return response()->api(['data' => $data])->header('Content-Type', self::CONTENT_TYPE); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -52,8 +52,8 @@ class CronController extends Controller | ||||
|         if (true === config('cer.download_enabled')) { | ||||
|             $return['exchange_rates'] = $this->exchangeRatesCronJob($config['force'], $config['date']); | ||||
|         } | ||||
|         $return['bill_warnings']          = $this->billWarningCronJob($config['force'], $config['date']); | ||||
|         $return['bill_notifications']     = $this->billWarningCronJob($config['force'], $config['date']); | ||||
| 
 | ||||
|         return response()->json($return); | ||||
|         return response()->api($return); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -32,7 +32,6 @@ use FireflyIII\Models\Preference; | ||||
| use FireflyIII\Transformers\PreferenceTransformer; | ||||
| use Illuminate\Http\JsonResponse; | ||||
| use Illuminate\Pagination\LengthAwarePaginator; | ||||
| use Illuminate\Support\Collection; | ||||
| use League\Fractal\Pagination\IlluminatePaginatorAdapter; | ||||
| use League\Fractal\Resource\Collection as FractalCollection; | ||||
| use League\Fractal\Resource\Item; | ||||
| @@ -86,7 +85,7 @@ class PreferencesController extends Controller | ||||
|         $manager     = $this->getManager(); | ||||
| 
 | ||||
|         if ('currencyPreference' === $preference->name) { | ||||
|             throw new FireflyException('Please use api/v1/currencies/default instead.'); | ||||
|             throw new FireflyException('Please use api/v1/currencies/native instead.'); | ||||
|         } | ||||
| 
 | ||||
|         /** @var PreferenceTransformer $transformer */ | ||||
| @@ -98,34 +97,6 @@ class PreferencesController extends Controller | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * TODO This endpoint is not documented. | ||||
|      * | ||||
|      * Return a single preference by name. | ||||
|      * | ||||
|      * @param Collection<int, Preference> $collection | ||||
|      */ | ||||
|     public function showList(Collection $collection): JsonResponse | ||||
|     { | ||||
|         $manager     = $this->getManager(); | ||||
|         $count       = $collection->count(); | ||||
|         $pageSize    = $this->parameters->get('limit'); | ||||
|         $preferences = $collection->slice(($this->parameters->get('page') - 1) * $pageSize, $pageSize); | ||||
| 
 | ||||
|         // make paginator:
 | ||||
|         $paginator   = new LengthAwarePaginator($preferences, $count, $pageSize, $this->parameters->get('page')); | ||||
|         $paginator->setPath(route('api.v1.preferences.show-list').$this->buildParams()); | ||||
| 
 | ||||
|         /** @var PreferenceTransformer $transformer */ | ||||
|         $transformer = app(PreferenceTransformer::class); | ||||
|         $transformer->setParameters($this->parameters); | ||||
| 
 | ||||
|         $resource    = new FractalCollection($preferences, $transformer, self::RESOURCE_KEY); | ||||
|         $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); | ||||
| 
 | ||||
|         return response()->json($manager->createData($resource)->toArray())->header('Content-Type', self::CONTENT_TYPE); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * This endpoint is documented at: | ||||
|      * https://api-docs.firefly-iii.org/?urls.primaryName=2.0.0%20(v1)#/preferences/storePreference
 | ||||
| @@ -161,7 +132,7 @@ class PreferencesController extends Controller | ||||
|     public function update(PreferenceUpdateRequest $request, Preference $preference): JsonResponse | ||||
|     { | ||||
|         if ('currencyPreference' === $preference->name) { | ||||
|             throw new FireflyException('Please use api/v1/currencies/default instead.'); | ||||
|             throw new FireflyException('Please use api/v1/currencies/native instead.'); | ||||
|         } | ||||
| 
 | ||||
|         $manager     = $this->getManager(); | ||||
|   | ||||
| @@ -58,6 +58,7 @@ class AutocompleteRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'date' => 'date|after:1900-01-01|before:2099-12-31', | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -24,7 +24,7 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Requests\Data; | ||||
| 
 | ||||
| use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Exceptions\ValidationException; | ||||
| use FireflyIII\Support\Request\ChecksLogin; | ||||
| use FireflyIII\Support\Request\ConvertsDataTypes; | ||||
| use Illuminate\Foundation\Http\FormRequest; | ||||
| @@ -49,12 +49,13 @@ class DateRequest extends FormRequest | ||||
|         $start->startOfDay(); | ||||
|         $end->endOfDay(); | ||||
|         if ($start->diffInYears($end, true) > 5) { | ||||
|             throw new FireflyException('Date range out of range.'); | ||||
|             throw new ValidationException('Date range out of range.'); | ||||
|         } | ||||
| 
 | ||||
|         return [ | ||||
|             'start' => $start, | ||||
|             'end'   => $end, | ||||
|             'date'  => $this->getCarbonDate('date'), | ||||
|         ]; | ||||
|     } | ||||
| 
 | ||||
| @@ -64,8 +65,9 @@ class DateRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'start' => 'required|date', | ||||
|             'end'   => 'required|date|after:start', | ||||
|             'date'  => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'start' => 'date|after:1900-01-01|before:2099-12-31|before:end|required_with:end', | ||||
|             'end'   => 'date|after:1900-01-01|before:2099-12-31|after:start|required_with:start', | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -26,6 +26,7 @@ namespace FireflyIII\Api\V1\Requests\Models\Account; | ||||
| 
 | ||||
| use FireflyIII\Models\Account; | ||||
| use FireflyIII\Models\Location; | ||||
| use FireflyIII\Repositories\Account\AccountRepositoryInterface; | ||||
| use FireflyIII\Rules\IsBoolean; | ||||
| use FireflyIII\Rules\UniqueAccountNumber; | ||||
| use FireflyIII\Rules\UniqueIban; | ||||
| @@ -33,6 +34,8 @@ use FireflyIII\Support\Request\AppendsLocationData; | ||||
| use FireflyIII\Support\Request\ChecksLogin; | ||||
| use FireflyIII\Support\Request\ConvertsDataTypes; | ||||
| use Illuminate\Foundation\Http\FormRequest; | ||||
| use Illuminate\Support\Facades\Log; | ||||
| use Illuminate\Validation\Validator; | ||||
| 
 | ||||
| /** | ||||
|  * Class UpdateRequest | ||||
| @@ -112,4 +115,36 @@ class UpdateRequest extends FormRequest | ||||
| 
 | ||||
|         return Location::requestRules($rules); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Configure the validator instance with special rules for after the basic validation rules. | ||||
|      */ | ||||
|     public function withValidator(Validator $validator): void | ||||
|     { | ||||
|         $validator->after( | ||||
|             function (Validator $validator): void { | ||||
|                 // validate start before end only if both are there.
 | ||||
|                 $data       = $validator->getData(); | ||||
| 
 | ||||
|                 /** @var Account $account */ | ||||
|                 $account    = $this->route()->parameter('account'); | ||||
| 
 | ||||
|                 /** @var AccountRepositoryInterface $repository */ | ||||
|                 $repository = app(AccountRepositoryInterface::class); | ||||
|                 $currency   = $repository->getAccountCurrency($account); | ||||
| 
 | ||||
|                 // how many piggies are attached?
 | ||||
|                 $piggyBanks = $account->piggyBanks()->count(); | ||||
|                 if ($piggyBanks > 0 && array_key_exists('currency_code', $data) && $data['currency_code'] !== $currency->code) { | ||||
|                     $validator->errors()->add('currency_code', (string) trans('validation.piggy_no_change_currency')); | ||||
|                 } | ||||
|                 if ($piggyBanks > 0 && array_key_exists('currency_id', $data) && (int) $data['currency_id'] !== $currency->id) { | ||||
|                     $validator->errors()->add('currency_id', (string) trans('validation.piggy_no_change_currency')); | ||||
|                 } | ||||
|             } | ||||
|         ); | ||||
|         if ($validator->fails()) { | ||||
|             Log::channel('audit')->error(sprintf('Validation errors in %s', __CLASS__), $validator->errors()->toArray()); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -66,8 +66,8 @@ class Request extends FormRequest | ||||
|             'currency_id'   => 'numeric|exists:transaction_currencies,id', | ||||
|             'currency_code' => 'min:3|max:51|exists:transaction_currencies,code', | ||||
|             'amount'        => ['nullable', new IsValidPositiveAmount()], | ||||
|             'start'         => 'date', | ||||
|             'end'           => 'date', | ||||
|             'start'         => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end'           => 'date|after:1900-01-01|before:2099-12-31', | ||||
|         ]; | ||||
|     } | ||||
| 
 | ||||
|   | ||||
| @@ -78,9 +78,9 @@ class StoreRequest extends FormRequest | ||||
|             'amount_max'     => ['required', new IsValidPositiveAmount()], | ||||
|             'currency_id'    => 'numeric|exists:transaction_currencies,id', | ||||
|             'currency_code'  => 'min:3|max:51|exists:transaction_currencies,code', | ||||
|             'date'           => 'date|required', | ||||
|             'end_date'       => 'nullable|date|after:date', | ||||
|             'extension_date' => 'nullable|date|after:date', | ||||
|             'date'           => 'date|required|after:1900-01-01|before:2099-12-31', | ||||
|             'end_date'       => 'nullable|date|after:date|after:1900-01-01|before:2099-12-31', | ||||
|             'extension_date' => 'nullable|date|after:date|after:1900-01-01|before:2099-12-31', | ||||
|             'repeat_freq'    => 'in:weekly,monthly,quarterly,half-year,yearly|required', | ||||
|             'skip'           => 'min:0|max:31|numeric', | ||||
|             'active'         => [new IsBoolean()], | ||||
| @@ -95,16 +95,40 @@ class StoreRequest extends FormRequest | ||||
|     { | ||||
|         $validator->after( | ||||
|             static function (Validator $validator): void { | ||||
|                 $data = $validator->getData(); | ||||
|                 $min  = (string) ($data['amount_min'] ?? '0'); | ||||
|                 $max  = (string) ($data['amount_max'] ?? '0'); | ||||
|                 $data   = $validator->getData(); | ||||
|                 $min    = $data['amount_min'] ?? '0'; | ||||
|                 $max    = $data['amount_max'] ?? '0'; | ||||
| 
 | ||||
|                 if (1 === bccomp($min, $max)) { | ||||
|                 if (is_array($min) || is_array($max)) { | ||||
|                     $validator->errors()->add('amount_min', (string) trans('validation.generic_invalid')); | ||||
|                     $validator->errors()->add('amount_max', (string) trans('validation.generic_invalid')); | ||||
|                     $min = '0'; | ||||
|                     $max = '0'; | ||||
|                 } | ||||
|                 $result = false; | ||||
| 
 | ||||
|                 try { | ||||
|                     $result = bccomp($min, $max); | ||||
|                 } catch (\ValueError $e) { | ||||
|                     Log::error($e->getMessage()); | ||||
|                     $validator->errors()->add('amount_min', (string) trans('validation.generic_invalid')); | ||||
|                     $validator->errors()->add('amount_max', (string) trans('validation.generic_invalid')); | ||||
|                 } | ||||
| 
 | ||||
|                 if (1 === $result) { | ||||
|                     $validator->errors()->add('amount_min', (string) trans('validation.amount_min_over_max')); | ||||
|                 } | ||||
|             } | ||||
|         ); | ||||
|         if ($validator->fails()) { | ||||
|         $failed = false; | ||||
| 
 | ||||
|         try { | ||||
|             $failed = $validator->fails(); | ||||
|         } catch (\TypeError $e) { | ||||
|             Log::error($e->getMessage()); | ||||
|             $failed = false; | ||||
|         } | ||||
|         if ($failed) { | ||||
|             Log::channel('audit')->error(sprintf('Validation errors in %s', __CLASS__), $validator->errors()->toArray()); | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -81,9 +81,9 @@ class UpdateRequest extends FormRequest | ||||
|             'amount_max'     => ['nullable', new IsValidPositiveAmount()], | ||||
|             'currency_id'    => 'numeric|exists:transaction_currencies,id', | ||||
|             'currency_code'  => 'min:3|max:51|exists:transaction_currencies,code', | ||||
|             'date'           => 'date', | ||||
|             'end_date'       => 'date|after:date', | ||||
|             'extension_date' => 'date|after:date', | ||||
|             'date'           => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end_date'       => 'date|after:date|after:1900-01-01|before:2099-12-31', | ||||
|             'extension_date' => 'date|after:date|after:1900-01-01|before:2099-12-31', | ||||
|             'repeat_freq'    => 'in:weekly,monthly,quarterly,half-year,yearly', | ||||
|             'skip'           => 'min:0|max:31|numeric', | ||||
|             'active'         => [new IsBoolean()], | ||||
|   | ||||
| @@ -67,8 +67,8 @@ class UpdateRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'start'         => 'date', | ||||
|             'end'           => 'date', | ||||
|             'start'         => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end'           => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'amount'        => ['nullable', new IsValidPositiveAmount()], | ||||
|             'currency_id'   => 'numeric|exists:transaction_currencies,id', | ||||
|             'currency_code' => 'min:3|max:51|exists:transaction_currencies,code', | ||||
|   | ||||
| @@ -2,7 +2,7 @@ | ||||
| 
 | ||||
| /* | ||||
|  * DestroyRequest.php | ||||
|  * Copyright (c) 2024 james@firefly-iii.org. | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -22,7 +22,7 @@ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Request\Model\ExchangeRate; | ||||
| namespace FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Support\Request\ChecksLogin; | ||||
| @@ -34,7 +34,7 @@ class DestroyRequest extends FormRequest | ||||
|     use ChecksLogin; | ||||
|     use ConvertsDataTypes; | ||||
| 
 | ||||
|     public function getDate(): Carbon | ||||
|     public function getDate(): ?Carbon | ||||
|     { | ||||
|         return $this->getCarbonDate('date'); | ||||
|     } | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * DestroyRequest.php | ||||
|  * Copyright (c) 2024 james@firefly-iii.org. | ||||
|  * StoreRequest.php | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -22,7 +22,7 @@ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Request\Model\ExchangeRate; | ||||
| namespace FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Models\TransactionCurrency; | ||||
| @@ -1,8 +1,8 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * DestroyRequest.php | ||||
|  * Copyright (c) 2024 james@firefly-iii.org. | ||||
|  * UpdateRequest.php | ||||
|  * Copyright (c) 2025 james@firefly-iii.org. | ||||
|  * | ||||
|  * This file is part of Firefly III (https://github.com/firefly-iii). | ||||
|  * | ||||
| @@ -22,7 +22,7 @@ | ||||
| 
 | ||||
| declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V2\Request\Model\ExchangeRate; | ||||
| namespace FireflyIII\Api\V1\Requests\Models\CurrencyExchangeRate; | ||||
| 
 | ||||
| use Carbon\Carbon; | ||||
| use FireflyIII\Support\Request\ChecksLogin; | ||||
| @@ -154,7 +154,7 @@ class UpdateRequest extends FormRequest | ||||
|         return [ | ||||
|             'title'                                => sprintf('min:1|max:255|uniqueObjectForUser:recurrences,title,%d', $recurrence->id), | ||||
|             'description'                          => 'min:1|max:32768', | ||||
|             'first_date'                           => 'date', | ||||
|             'first_date'                           => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'apply_rules'                          => [new IsBoolean()], | ||||
|             'active'                               => [new IsBoolean()], | ||||
|             'repeat_until'                         => 'nullable|date', | ||||
|   | ||||
| @@ -71,8 +71,8 @@ class TestRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'start'      => 'date', | ||||
|             'end'        => 'date|after_or_equal:start', | ||||
|             'start'      => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end'        => 'date|after_or_equal:start|after:1900-01-01|before:2099-12-31', | ||||
|             'accounts'   => '', | ||||
|             'accounts.*' => 'required|exists:accounts,id|belongsToUser:accounts', | ||||
|         ]; | ||||
|   | ||||
| @@ -65,8 +65,8 @@ class TriggerRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'start'      => 'date', | ||||
|             'end'        => 'date|after_or_equal:start', | ||||
|             'start'      => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end'        => 'date|after_or_equal:start|after:1900-01-01|before:2099-12-31', | ||||
|             'accounts'   => '', | ||||
|             'accounts.*' => 'exists:accounts,id|belongsToUser:accounts', | ||||
|         ]; | ||||
|   | ||||
| @@ -65,8 +65,8 @@ class TestRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'start'      => 'date', | ||||
|             'end'        => 'date|after_or_equal:start', | ||||
|             'start'      => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end'        => 'date|after_or_equal:start|after:1900-01-01|before:2099-12-31', | ||||
|             'accounts'   => '', | ||||
|             'accounts.*' => 'exists:accounts,id|belongsToUser:accounts', | ||||
|         ]; | ||||
|   | ||||
| @@ -69,8 +69,8 @@ class TriggerRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'start' => 'date', | ||||
|             'end'   => 'date|after_or_equal:start', | ||||
|             'start' => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end'   => 'date|after_or_equal:start|after:1900-01-01|before:2099-12-31', | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -62,7 +62,7 @@ class StoreRequest extends FormRequest | ||||
|         $rules = [ | ||||
|             'tag'         => 'required|min:1|uniqueObjectForUser:tags,tag|max:1024', | ||||
|             'description' => 'min:1|nullable|max:32768', | ||||
|             'date'        => 'date|nullable', | ||||
|             'date'        => 'date|nullable|after:1900-01-01|before:2099-12-31', | ||||
|         ]; | ||||
| 
 | ||||
|         return Location::requestRules($rules); | ||||
|   | ||||
| @@ -63,11 +63,10 @@ class UpdateRequest extends FormRequest | ||||
|     { | ||||
|         /** @var Tag $tag */ | ||||
|         $tag   = $this->route()->parameter('tagOrId'); | ||||
|         // TODO check if uniqueObjectForUser is obsolete
 | ||||
|         $rules = [ | ||||
|             'tag'         => 'min:1|max:1024|uniqueObjectForUser:tags,tag,'.$tag->id, | ||||
|             'description' => 'min:1|nullable|max:32768', | ||||
|             'date'        => 'date|nullable', | ||||
|             'date'        => 'date|nullable|after:1900-01-01|before:2099-12-31', | ||||
|         ]; | ||||
| 
 | ||||
|         return Location::requestRules($rules); | ||||
|   | ||||
| @@ -24,6 +24,7 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Api\V1\Requests\Models\Transaction; | ||||
| 
 | ||||
| use FireflyIII\Models\Location; | ||||
| use FireflyIII\Rules\BelongsUser; | ||||
| use FireflyIII\Rules\IsBoolean; | ||||
| use FireflyIII\Rules\IsDateOrTime; | ||||
| @@ -82,82 +83,87 @@ class StoreRequest extends FormRequest | ||||
|         foreach ($this->get('transactions') as $transaction) { | ||||
|             $object   = new NullArrayObject($transaction); | ||||
|             $return[] = [ | ||||
|                 'type'                  => $this->clearString($object['type']), | ||||
|                 'date'                  => $this->dateFromValue($object['date']), | ||||
|                 'order'                 => $this->integerFromValue((string) $object['order']), | ||||
|                 'type'                    => $this->clearString($object['type']), | ||||
|                 'date'                    => $this->dateFromValue($object['date']), | ||||
|                 'order'                   => $this->integerFromValue((string) $object['order']), | ||||
| 
 | ||||
|                 'currency_id'           => $this->integerFromValue((string) $object['currency_id']), | ||||
|                 'currency_code'         => $this->clearString((string) $object['currency_code']), | ||||
|                 'currency_id'             => $this->integerFromValue((string) $object['currency_id']), | ||||
|                 'currency_code'           => $this->clearString((string) $object['currency_code']), | ||||
| 
 | ||||
|                 // location
 | ||||
|                 'latitude'                => $this->floatFromValue((string) $object['latitude']), | ||||
|                 'longitude'               => $this->floatFromValue((string) $object['longitude']), | ||||
|                 'zoom_level'              => $this->integerFromValue((string) $object['zoom_level']), | ||||
| 
 | ||||
|                 // foreign currency info:
 | ||||
|                 'foreign_currency_id'   => $this->integerFromValue((string) $object['foreign_currency_id']), | ||||
|                 'foreign_currency_code' => $this->clearString((string) $object['foreign_currency_code']), | ||||
|                 'foreign_currency_id'     => $this->integerFromValue((string) $object['foreign_currency_id']), | ||||
|                 'foreign_currency_code'   => $this->clearString((string) $object['foreign_currency_code']), | ||||
| 
 | ||||
|                 // amount and foreign amount. Cannot be 0.
 | ||||
|                 'amount'                => $this->clearString((string) $object['amount']), | ||||
|                 'foreign_amount'        => $this->clearString((string) $object['foreign_amount']), | ||||
|                 'amount'                  => $this->clearString((string) $object['amount']), | ||||
|                 'foreign_amount'          => $this->clearString((string) $object['foreign_amount']), | ||||
| 
 | ||||
|                 // description.
 | ||||
|                 'description'           => $this->clearString($object['description']), | ||||
|                 'description'             => $this->clearString($object['description']), | ||||
| 
 | ||||
|                 // source of transaction. If everything is null, assume cash account.
 | ||||
|                 'source_id'             => $this->integerFromValue((string) $object['source_id']), | ||||
|                 'source_name'           => $this->clearString((string) $object['source_name']), | ||||
|                 'source_iban'           => $this->clearIban((string) $object['source_iban']), | ||||
|                 'source_number'         => $this->clearString((string) $object['source_number']), | ||||
|                 'source_bic'            => $this->clearString((string) $object['source_bic']), | ||||
|                 'source_id'               => $this->integerFromValue((string) $object['source_id']), | ||||
|                 'source_name'             => $this->clearString((string) $object['source_name']), | ||||
|                 'source_iban'             => $this->clearIban((string) $object['source_iban']), | ||||
|                 'source_number'           => $this->clearString((string) $object['source_number']), | ||||
|                 'source_bic'              => $this->clearString((string) $object['source_bic']), | ||||
| 
 | ||||
|                 // destination of transaction. If everything is null, assume cash account.
 | ||||
|                 'destination_id'        => $this->integerFromValue((string) $object['destination_id']), | ||||
|                 'destination_name'      => $this->clearString((string) $object['destination_name']), | ||||
|                 'destination_iban'      => $this->clearIban((string) $object['destination_iban']), | ||||
|                 'destination_number'    => $this->clearString((string) $object['destination_number']), | ||||
|                 'destination_bic'       => $this->clearString((string) $object['destination_bic']), | ||||
|                 'destination_id'          => $this->integerFromValue((string) $object['destination_id']), | ||||
|                 'destination_name'        => $this->clearString((string) $object['destination_name']), | ||||
|                 'destination_iban'        => $this->clearIban((string) $object['destination_iban']), | ||||
|                 'destination_number'      => $this->clearString((string) $object['destination_number']), | ||||
|                 'destination_bic'         => $this->clearString((string) $object['destination_bic']), | ||||
| 
 | ||||
|                 // budget info
 | ||||
|                 'budget_id'             => $this->integerFromValue((string) $object['budget_id']), | ||||
|                 'budget_name'           => $this->clearString((string) $object['budget_name']), | ||||
|                 'budget_id'               => $this->integerFromValue((string) $object['budget_id']), | ||||
|                 'budget_name'             => $this->clearString((string) $object['budget_name']), | ||||
| 
 | ||||
|                 // category info
 | ||||
|                 'category_id'           => $this->integerFromValue((string) $object['category_id']), | ||||
|                 'category_name'         => $this->clearString((string) $object['category_name']), | ||||
|                 'category_id'             => $this->integerFromValue((string) $object['category_id']), | ||||
|                 'category_name'           => $this->clearString((string) $object['category_name']), | ||||
| 
 | ||||
|                 // journal bill reference. Optional. Will only work for withdrawals
 | ||||
|                 'bill_id'               => $this->integerFromValue((string) $object['bill_id']), | ||||
|                 'bill_name'             => $this->clearString((string) $object['bill_name']), | ||||
|                 'bill_id'                 => $this->integerFromValue((string) $object['bill_id']), | ||||
|                 'bill_name'               => $this->clearString((string) $object['bill_name']), | ||||
| 
 | ||||
|                 // piggy bank reference. Optional. Will only work for transfers
 | ||||
|                 'piggy_bank_id'         => $this->integerFromValue((string) $object['piggy_bank_id']), | ||||
|                 'piggy_bank_name'       => $this->clearString((string) $object['piggy_bank_name']), | ||||
|                 'piggy_bank_id'           => $this->integerFromValue((string) $object['piggy_bank_id']), | ||||
|                 'piggy_bank_name'         => $this->clearString((string) $object['piggy_bank_name']), | ||||
| 
 | ||||
|                 // some other interesting properties
 | ||||
|                 'reconciled'            => $this->convertBoolean((string) $object['reconciled']), | ||||
|                 'notes'                 => $this->clearStringKeepNewlines((string) $object['notes']), | ||||
|                 'tags'                  => $this->arrayFromValue($object['tags']), | ||||
|                 'reconciled'              => $this->convertBoolean((string) $object['reconciled']), | ||||
|                 'notes'                   => $this->clearStringKeepNewlines((string) $object['notes']), | ||||
|                 'tags'                    => $this->arrayFromValue($object['tags']), | ||||
| 
 | ||||
|                 // all custom fields:
 | ||||
|                 'internal_reference'    => $this->clearString((string) $object['internal_reference']), | ||||
|                 'external_id'           => $this->clearString((string) $object['external_id']), | ||||
|                 'original_source'       => sprintf('ff3-v%s', config('firefly.version')), | ||||
|                 'recurrence_id'         => $this->integerFromValue($object['recurrence_id']), | ||||
|                 'bunq_payment_id'       => $this->clearString((string) $object['bunq_payment_id']), | ||||
|                 'external_url'          => $this->clearString((string) $object['external_url']), | ||||
|                 'internal_reference'      => $this->clearString((string) $object['internal_reference']), | ||||
|                 'external_id'             => $this->clearString((string) $object['external_id']), | ||||
|                 'original_source'         => sprintf('ff3-v%s', config('firefly.version')), | ||||
|                 'recurrence_id'           => $this->integerFromValue($object['recurrence_id']), | ||||
|                 'bunq_payment_id'         => $this->clearString((string) $object['bunq_payment_id']), | ||||
|                 'external_url'            => $this->clearString((string) $object['external_url']), | ||||
| 
 | ||||
|                 'sepa_cc'               => $this->clearString((string) $object['sepa_cc']), | ||||
|                 'sepa_ct_op'            => $this->clearString((string) $object['sepa_ct_op']), | ||||
|                 'sepa_ct_id'            => $this->clearString((string) $object['sepa_ct_id']), | ||||
|                 'sepa_db'               => $this->clearString((string) $object['sepa_db']), | ||||
|                 'sepa_country'          => $this->clearString((string) $object['sepa_country']), | ||||
|                 'sepa_ep'               => $this->clearString((string) $object['sepa_ep']), | ||||
|                 'sepa_ci'               => $this->clearString((string) $object['sepa_ci']), | ||||
|                 'sepa_batch_id'         => $this->clearString((string) $object['sepa_batch_id']), | ||||
|                 'sepa_cc'                 => $this->clearString((string) $object['sepa_cc']), | ||||
|                 'sepa_ct_op'              => $this->clearString((string) $object['sepa_ct_op']), | ||||
|                 'sepa_ct_id'              => $this->clearString((string) $object['sepa_ct_id']), | ||||
|                 'sepa_db'                 => $this->clearString((string) $object['sepa_db']), | ||||
|                 'sepa_country'            => $this->clearString((string) $object['sepa_country']), | ||||
|                 'sepa_ep'                 => $this->clearString((string) $object['sepa_ep']), | ||||
|                 'sepa_ci'                 => $this->clearString((string) $object['sepa_ci']), | ||||
|                 'sepa_batch_id'           => $this->clearString((string) $object['sepa_batch_id']), | ||||
|                 // custom date fields. Must be Carbon objects. Presence is optional.
 | ||||
|                 'interest_date'         => $this->dateFromValue($object['interest_date']), | ||||
|                 'book_date'             => $this->dateFromValue($object['book_date']), | ||||
|                 'process_date'          => $this->dateFromValue($object['process_date']), | ||||
|                 'due_date'              => $this->dateFromValue($object['due_date']), | ||||
|                 'payment_date'          => $this->dateFromValue($object['payment_date']), | ||||
|                 'invoice_date'          => $this->dateFromValue($object['invoice_date']), | ||||
|                 'interest_date'           => $this->dateFromValue($object['interest_date']), | ||||
|                 'book_date'               => $this->dateFromValue($object['book_date']), | ||||
|                 'process_date'            => $this->dateFromValue($object['process_date']), | ||||
|                 'due_date'                => $this->dateFromValue($object['due_date']), | ||||
|                 'payment_date'            => $this->dateFromValue($object['payment_date']), | ||||
|                 'invoice_date'            => $this->dateFromValue($object['invoice_date']), | ||||
|             ]; | ||||
|         } | ||||
| 
 | ||||
| @@ -171,6 +177,7 @@ class StoreRequest extends FormRequest | ||||
|     { | ||||
|         app('log')->debug('Collect rules of TransactionStoreRequest'); | ||||
|         $validProtocols = config('firefly.valid_url_protocols'); | ||||
|         $locationRules  = Location::requestRules([]); | ||||
| 
 | ||||
|         return [ | ||||
|             // basic fields for group:
 | ||||
| @@ -178,6 +185,11 @@ class StoreRequest extends FormRequest | ||||
|             'error_if_duplicate_hash'              => [new IsBoolean()], | ||||
|             'apply_rules'                          => [new IsBoolean()], | ||||
| 
 | ||||
|             // location rules
 | ||||
|             'transactions.*.latitude'              => $locationRules['latitude'], | ||||
|             'transactions.*.longitude'             => $locationRules['longitude'], | ||||
|             'transactions.*.zoom_level'            => $locationRules['zoom_level'], | ||||
| 
 | ||||
|             // transaction rules (in array for splits):
 | ||||
|             'transactions.*.type'                  => 'required|in:withdrawal,deposit,transfer,opening-balance,reconciliation', | ||||
|             'transactions.*.date'                  => ['required', new IsDateOrTime()], | ||||
|   | ||||
							
								
								
									
										65
									
								
								app/Api/V1/Requests/Models/UserGroup/UpdateRequest.php
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								app/Api/V1/Requests/Models/UserGroup/UpdateRequest.php
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,65 @@ | ||||
| <?php | ||||
| 
 | ||||
| /* | ||||
|  * UpdateRequest.php | ||||
|  * Copyright (c) 2021 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\V1\Requests\Models\UserGroup; | ||||
| 
 | ||||
| use FireflyIII\Models\UserGroup; | ||||
| use FireflyIII\Support\Request\ChecksLogin; | ||||
| use FireflyIII\Support\Request\ConvertsDataTypes; | ||||
| use Illuminate\Foundation\Http\FormRequest; | ||||
| 
 | ||||
| /** | ||||
|  * Class UpdateRequest | ||||
|  */ | ||||
| class UpdateRequest extends FormRequest | ||||
| { | ||||
|     use ChecksLogin; | ||||
|     use ConvertsDataTypes; | ||||
| 
 | ||||
|     public function getData(): array | ||||
|     { | ||||
|         $fields = [ | ||||
|             'title'                => ['title', 'convertString'], | ||||
|             'native_currency_id'   => ['native_currency_id', 'convertInteger'], | ||||
|             'native_currency_code' => ['native_currency_code', 'convertString'], | ||||
|         ]; | ||||
| 
 | ||||
|         return $this->getAllData($fields); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * Rules for this request. | ||||
|      */ | ||||
|     public function rules(): array | ||||
|     { | ||||
|         /** @var UserGroup $userGroup */ | ||||
|         $userGroup = $this->route()->parameter('userGroup'); | ||||
| 
 | ||||
|         return [ | ||||
|             'title'                => ['required', 'min:1', 'max:255'], | ||||
|             'native_currency_id'   => 'exists:transaction_currencies,id', | ||||
|             'native_currency_code' => 'exists:transaction_currencies,code', | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
| @@ -57,6 +57,10 @@ class CronRequest extends FormRequest | ||||
|         if ($this->has('date')) { | ||||
|             $data['date'] = $this->getCarbonDate('date'); | ||||
|         } | ||||
|         // catch NULL.
 | ||||
|         if (null === $data['date']) { | ||||
|             $data['date'] = today(config('app.timezone')); | ||||
|         } | ||||
| 
 | ||||
|         return $data; | ||||
|     } | ||||
| @@ -68,7 +72,7 @@ class CronRequest extends FormRequest | ||||
|     { | ||||
|         return [ | ||||
|             'force' => 'in:true,false', | ||||
|             'date'  => 'date', | ||||
|             'date'  => 'nullable|date|after:1900-01-01|before:2099-12-31', | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -55,7 +55,7 @@ class AccountController extends Controller | ||||
|                 $userGroup        = $this->validateUserGroup($request); | ||||
|                 $this->repository = app(AccountRepositoryInterface::class); | ||||
|                 $this->repository->setUserGroup($userGroup); | ||||
|                 $this->default    = app('amount')->getDefaultCurrency(); | ||||
|                 $this->default    = app('amount')->getNativeCurrency(); | ||||
|                 $this->converter  = app(ExchangeRateConverter::class); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|   | ||||
| @@ -57,7 +57,7 @@ class AccountController extends Controller | ||||
|                 $this->repository = app(AccountRepositoryInterface::class); | ||||
|                 $this->repository->setUserGroup($this->validateUserGroup($request)); | ||||
|                 $this->chartData  = new ChartData(); | ||||
|                 $this->default    = app('amount')->getDefaultCurrency(); | ||||
|                 $this->default    = app('amount')->getNativeCurrency(); | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
|   | ||||
| @@ -62,7 +62,7 @@ class BalanceController extends Controller | ||||
|                 $this->repository->setUserGroup($userGroup); | ||||
|                 $this->collector->setUserGroup($userGroup); | ||||
|                 $this->chartData  = new ChartData(); | ||||
|                 // $this->default    = app('amount')->getDefaultCurrency();
 | ||||
|                 // $this->default    = app('amount')->getNativeCurrency();
 | ||||
| 
 | ||||
|                 return $next($request); | ||||
|             } | ||||
| @@ -87,7 +87,7 @@ class BalanceController extends Controller | ||||
| 
 | ||||
|         // prepare for currency conversion and data collection:
 | ||||
|         /** @var TransactionCurrency $default */ | ||||
|         $default         = app('amount')->getDefaultCurrency(); | ||||
|         $default         = app('amount')->getNativeCurrency(); | ||||
| 
 | ||||
|         // get journals for entire period:
 | ||||
| 
 | ||||
|   | ||||
| @@ -63,7 +63,7 @@ class BudgetController extends Controller | ||||
|                 $this->repository    = app(BudgetRepositoryInterface::class); | ||||
|                 $this->blRepository  = app(BudgetLimitRepositoryInterface::class); | ||||
|                 $this->opsRepository = app(OperationsRepositoryInterface::class); | ||||
|                 $this->currency      = app('amount')->getDefaultCurrency(); | ||||
|                 $this->currency      = app('amount')->getNativeCurrency(); | ||||
|                 $userGroup           = $this->validateUserGroup($request); | ||||
|                 $this->repository->setUserGroup($userGroup); | ||||
|                 $this->opsRepository->setUserGroup($userGroup); | ||||
|   | ||||
| @@ -82,7 +82,7 @@ class CategoryController extends Controller | ||||
|         /** @var Carbon $end */ | ||||
|         $end        = $this->parameters->get('end'); | ||||
|         $accounts   = $this->accountRepos->getAccountsByType([AccountTypeEnum::DEBT->value, AccountTypeEnum::LOAN->value, AccountTypeEnum::MORTGAGE->value, AccountTypeEnum::ASSET->value, AccountTypeEnum::DEFAULT->value]); | ||||
|         $default    = app('amount')->getDefaultCurrency(); | ||||
|         $default    = app('amount')->getNativeCurrency(); | ||||
|         $converter  = new ExchangeRateConverter(); | ||||
|         $currencies = []; | ||||
|         $return     = []; | ||||
|   | ||||
| @@ -66,7 +66,7 @@ class ShowController extends Controller | ||||
|             $default = 1 === $group->pivot->group_default; | ||||
|         } | ||||
|         $currency->userGroupEnabled = $enabled; | ||||
|         $currency->userGroupDefault = $default; | ||||
|         $currency->userGroupNative  = $default; | ||||
| 
 | ||||
| 
 | ||||
|         $transformer                = new CurrencyTransformer(); | ||||
|   | ||||
| @@ -118,7 +118,7 @@ class BasicController extends Controller | ||||
|     private function getBalanceInformation(Carbon $start, Carbon $end): array | ||||
|     { | ||||
|         $object    = new SummaryBalanceGrouped(); | ||||
|         $default   = app('amount')->getDefaultCurrency(); | ||||
|         $default   = app('amount')->getNativeCurrency(); | ||||
| 
 | ||||
|         $object->setDefault($default); | ||||
| 
 | ||||
| @@ -233,7 +233,7 @@ class BasicController extends Controller | ||||
|         $available    = $this->abRepository->getAvailableBudgetWithCurrency($start, $end); | ||||
|         $budgets      = $this->budgetRepository->getActiveBudgets(); | ||||
|         $spent        = $this->opsRepository->listExpenses($start, $end, null, $budgets); | ||||
|         $default      = app('amount')->getDefaultCurrency(); | ||||
|         $default      = app('amount')->getNativeCurrency(); | ||||
|         $currencies   = []; | ||||
|         $converter    = new ExchangeRateConverter(); | ||||
| 
 | ||||
|   | ||||
| @@ -109,8 +109,8 @@ class InfiniteListRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'start'     => 'date', | ||||
|             'end'       => 'date|after:start', | ||||
|             'start'     => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end'       => 'date|after:start|after:1900-01-01|before:2099-12-31', | ||||
|             'start_row' => 'integer|min:0|max:4294967296', | ||||
|             'end_row'   => 'integer|min:0|max:4294967296|gt:start_row', | ||||
|         ]; | ||||
|   | ||||
| @@ -84,8 +84,8 @@ class ListRequest extends FormRequest | ||||
|     public function rules(): array | ||||
|     { | ||||
|         return [ | ||||
|             'start' => 'date', | ||||
|             'end'   => 'date|after:start', | ||||
|             'start' => 'date|after:1900-01-01|before:2099-12-31', | ||||
|             'end'   => 'date|after:start|after:1900-01-01|before:2099-12-31', | ||||
|         ]; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -63,7 +63,7 @@ class CorrectsCurrencies extends Command | ||||
|         $repos           = app(CurrencyRepositoryInterface::class); | ||||
| 
 | ||||
|         // first check if the user has any default currency (not necessarily the case, so can be forced).
 | ||||
|         $defaultCurrency = app('amount')->getDefaultCurrencyByUserGroup($userGroup); | ||||
|         $defaultCurrency = app('amount')->getNativeCurrencyByUserGroup($userGroup); | ||||
| 
 | ||||
|         Log::debug(sprintf('Now correcting currencies for user group #%d', $userGroup->id)); | ||||
|         $found           = [$defaultCurrency->id]; | ||||
|   | ||||
| @@ -88,7 +88,7 @@ class CorrectsNativeAmounts extends Command | ||||
| 
 | ||||
|         // do a check with the group's currency so we can skip some stuff.
 | ||||
|         Preferences::mark(); | ||||
|         $currency = app('amount')->getDefaultCurrencyByUserGroup($userGroup); | ||||
|         $currency = app('amount')->getNativeCurrencyByUserGroup($userGroup); | ||||
| 
 | ||||
|         $this->recalculatePiggyBanks($userGroup, $currency); | ||||
|         $this->recalculateBudgets($userGroup, $currency); | ||||
|   | ||||
| @@ -132,6 +132,6 @@ class CorrectsOpeningBalanceCurrencies extends Command | ||||
|         $repos = app(AccountRepositoryInterface::class); | ||||
|         $repos->setUser($account->user); | ||||
| 
 | ||||
|         return $repos->getAccountCurrency($account) ?? app('amount')->getDefaultCurrencyByUserGroup($account->userGroup); | ||||
|         return $repos->getAccountCurrency($account) ?? app('amount')->getNativeCurrencyByUserGroup($account->userGroup); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -38,7 +38,7 @@ class CreatesDatabase extends Command | ||||
| 
 | ||||
|     public function handle(): int | ||||
|     { | ||||
|         if ('mysql' !== env('DB_CONNECTION')) { | ||||
|         if ('mysql' !== env('DB_CONNECTION')) { // @phpstan-ignore larastan.noEnvCallsOutsideOfConfig */
 | ||||
|             $this->friendlyInfo(sprintf('CreateDB does not apply to "%s", skipped.', env('DB_CONNECTION'))); | ||||
| 
 | ||||
|             return 0; | ||||
| @@ -60,7 +60,7 @@ class CreatesDatabase extends Command | ||||
| 
 | ||||
|         // when it fails, display error
 | ||||
|         try { | ||||
|             $pdo = new \PDO($dsn, env('DB_USERNAME'), env('DB_PASSWORD'), $options); | ||||
|             $pdo = new \PDO($dsn, (string) env('DB_USERNAME'), (string) env('DB_PASSWORD'), $options); | ||||
|         } catch (\PDOException $e) { | ||||
|             $this->friendlyError(sprintf('Error when connecting to DB: %s', $e->getMessage())); | ||||
| 
 | ||||
|   | ||||
| @@ -107,7 +107,7 @@ class UpgradesAccountCurrencies extends Command | ||||
|         $accounts        = $this->accountRepos->getAccountsByType([AccountTypeEnum::DEFAULT->value, AccountTypeEnum::ASSET->value]); | ||||
| 
 | ||||
|         // get user's currency preference:
 | ||||
|         $defaultCurrency = app('amount')->getDefaultCurrencyByUserGroup($user->userGroup); | ||||
|         $defaultCurrency = app('amount')->getNativeCurrencyByUserGroup($user->userGroup); | ||||
| 
 | ||||
|         /** @var Account $account */ | ||||
|         foreach ($accounts as $account) { | ||||
|   | ||||
| @@ -66,7 +66,7 @@ class UpgradesBudgetLimits extends Command | ||||
|                     /** @var null|User $user */ | ||||
|                     $user = $budget->user; | ||||
|                     if (null !== $user) { | ||||
|                         $currency                             = app('amount')->getDefaultCurrencyByUserGroup($user->userGroup); | ||||
|                         $currency                             = app('amount')->getNativeCurrencyByUserGroup($user->userGroup); | ||||
|                         $budgetLimit->transaction_currency_id = $currency->id; | ||||
|                         $budgetLimit->save(); | ||||
|                         $this->friendlyInfo( | ||||
|   | ||||
| @@ -90,7 +90,7 @@ class UpgradesMultiPiggyBanks extends Command | ||||
|         $this->repository->setUser($piggyBank->account->user); | ||||
|         $this->accountRepository->setUser($piggyBank->account->user); | ||||
|         $repetition                         = $this->repository->getRepetition($piggyBank, true); | ||||
|         $currency                           = $this->accountRepository->getAccountCurrency($piggyBank->account) ?? app('amount')->getDefaultCurrencyByUserGroup($piggyBank->account->user->userGroup); | ||||
|         $currency                           = $this->accountRepository->getAccountCurrency($piggyBank->account) ?? app('amount')->getNativeCurrencyByUserGroup($piggyBank->account->user->userGroup); | ||||
| 
 | ||||
|         // update piggy bank to have a currency.
 | ||||
|         $piggyBank->transaction_currency_id = $currency->id; | ||||
|   | ||||
| @@ -24,6 +24,7 @@ declare(strict_types=1); | ||||
| 
 | ||||
| namespace FireflyIII\Exceptions; | ||||
| 
 | ||||
| use Brick\Math\Exception\NumberFormatException; | ||||
| use FireflyIII\Jobs\MailError; | ||||
| use Illuminate\Auth\Access\AuthorizationException; | ||||
| use Illuminate\Auth\AuthenticationException; | ||||
| @@ -124,7 +125,7 @@ class Handler extends ExceptionHandler | ||||
|         if ($e instanceof BadRequestHttpException) { | ||||
|             app('log')->debug('Return JSON BadRequestHttpException.'); | ||||
| 
 | ||||
|             return response()->json(['message' => $e->getMessage(), 'exception' => 'BadRequestHttpException'], 400); | ||||
|             return response()->json(['message' => $e->getMessage(), 'exception' => 'HttpException'], 400); | ||||
|         } | ||||
| 
 | ||||
|         if ($e instanceof BadHttpHeaderException) { | ||||
| @@ -133,6 +134,14 @@ class Handler extends ExceptionHandler | ||||
| 
 | ||||
|             return response()->json(['message' => $e->getMessage(), 'exception' => 'BadHttpHeaderException'], $e->statusCode); | ||||
|         } | ||||
|         if (($e instanceof ValidationException || $e instanceof NumberFormatException) && $expectsJson) { | ||||
|             $errorCode = 422; | ||||
| 
 | ||||
|             return response()->json( | ||||
|                 ['message' => sprintf('Validation exception: %s', $e->getMessage()), 'errors' => ['field' => 'Field is invalid']], | ||||
|                 $errorCode | ||||
|             ); | ||||
|         } | ||||
| 
 | ||||
|         if ($expectsJson) { | ||||
|             $errorCode = 500; | ||||
| @@ -156,7 +165,7 @@ class Handler extends ExceptionHandler | ||||
|             app('log')->debug(sprintf('Return JSON %s.', get_class($e))); | ||||
| 
 | ||||
|             return response()->json( | ||||
|                 ['message' => sprintf('Internal Firefly III Exception: %s', $e->getMessage()), 'exception' => get_class($e)], | ||||
|                 ['message' => sprintf('Internal Firefly III Exception: %s', $e->getMessage()), 'exception' => 'UndisclosedException'], | ||||
|                 $errorCode | ||||
|             ); | ||||
|         } | ||||
|   | ||||
| @@ -49,7 +49,7 @@ class BillFactory | ||||
|         app('log')->debug(sprintf('Now in %s', __METHOD__), $data); | ||||
|         $factory          = app(TransactionCurrencyFactory::class); | ||||
|         $currency         = $factory->find((int) ($data['currency_id'] ?? null), (string) ($data['currency_code'] ?? null)) ?? | ||||
|                     app('amount')->getDefaultCurrencyByUserGroup($this->user->userGroup); | ||||
|                     app('amount')->getNativeCurrencyByUserGroup($this->user->userGroup); | ||||
| 
 | ||||
|         try { | ||||
|             $skip   = array_key_exists('skip', $data) ? $data['skip'] : 0; | ||||
|   | ||||
| @@ -121,7 +121,7 @@ class PiggyBankFactory | ||||
|     private function getCurrency(array $data): TransactionCurrency | ||||
|     { | ||||
|         // currency:
 | ||||
|         $defaultCurrency = app('amount')->getDefaultCurrency(); | ||||
|         $defaultCurrency = app('amount')->getNativeCurrency(); | ||||
|         $currency        = null; | ||||
|         if (array_key_exists('transaction_currency_code', $data)) { | ||||
|             $currency = $this->currencyRepository->findByCode((string) ($data['transaction_currency_code'] ?? '')); | ||||
|   | ||||
| @@ -466,7 +466,7 @@ class TransactionJournalFactory | ||||
|         $preference = $this->accountRepository->getAccountCurrency($account); | ||||
|         if (null === $preference && null === $currency) { | ||||
|             // return user's default:
 | ||||
|             return app('amount')->getDefaultCurrencyByUserGroup($this->user->userGroup); | ||||
|             return app('amount')->getNativeCurrencyByUserGroup($this->user->userGroup); | ||||
|         } | ||||
|         $result     = $preference ?? $currency; | ||||
|         app('log')->debug(sprintf('Currency is now #%d (%s) because of account #%d (%s)', $result->id, $result->code, $account->id, $account->name)); | ||||
| @@ -576,7 +576,7 @@ class TransactionJournalFactory | ||||
| 
 | ||||
|     private function storeLocation(TransactionJournal $journal, NullArrayObject $data): void | ||||
|     { | ||||
|         if (true === $data['store_location']) { | ||||
|         if (null !== $data['longitude'] && null !== $data['latitude'] && null !== $data['zoom_level']) { | ||||
|             $location             = new Location(); | ||||
|             $location->longitude  = $data['longitude']; | ||||
|             $location->latitude   = $data['latitude']; | ||||
|   | ||||
| @@ -135,7 +135,7 @@ class MonthReportGenerator implements ReportGeneratorInterface | ||||
|         $journals          = array_reverse($journals, true); | ||||
|         $dayBeforeBalance  = Steam::finalAccountBalance($account, $date); | ||||
|         $startBalance      = $dayBeforeBalance['balance']; | ||||
|         $defaultCurrency   = app('amount')->getDefaultCurrencyByUserGroup($account->user->userGroup); | ||||
|         $defaultCurrency   = app('amount')->getNativeCurrencyByUserGroup($account->user->userGroup); | ||||
|         $currency          = $accountRepository->getAccountCurrency($account) ?? $defaultCurrency; | ||||
| 
 | ||||
|         foreach ($journals as $index => $journal) { | ||||
|   | ||||
| @@ -47,7 +47,7 @@ class AccountObserver | ||||
|         if (!Amount::convertToNative($account->user)) { | ||||
|             return; | ||||
|         } | ||||
|         $userCurrency = app('amount')->getDefaultCurrencyByUserGroup($account->user->userGroup); | ||||
|         $userCurrency = app('amount')->getNativeCurrencyByUserGroup($account->user->userGroup); | ||||
|         $repository   = app(AccountRepositoryInterface::class); | ||||
|         $currency     = $repository->getAccountCurrency($account); | ||||
|         if (null !== $currency && $currency->id !== $userCurrency->id && '' !== (string) $account->virtual_balance && 0 !== bccomp($account->virtual_balance, '0')) { | ||||
|   | ||||
| @@ -48,7 +48,7 @@ class AutoBudgetObserver | ||||
|         if (!Amount::convertToNative($autoBudget->budget->user)) { | ||||
|             return; | ||||
|         } | ||||
|         $userCurrency              = app('amount')->getDefaultCurrencyByUserGroup($autoBudget->budget->user->userGroup); | ||||
|         $userCurrency              = app('amount')->getNativeCurrencyByUserGroup($autoBudget->budget->user->userGroup); | ||||
|         $autoBudget->native_amount = null; | ||||
|         if ($autoBudget->transactionCurrency->id !== $userCurrency->id) { | ||||
|             $converter                 = new ExchangeRateConverter(); | ||||
|   | ||||
| @@ -50,7 +50,7 @@ class AvailableBudgetObserver | ||||
| 
 | ||||
|             return; | ||||
|         } | ||||
|         $userCurrency                   = app('amount')->getDefaultCurrencyByUserGroup($availableBudget->user->userGroup); | ||||
|         $userCurrency                   = app('amount')->getNativeCurrencyByUserGroup($availableBudget->user->userGroup); | ||||
|         $availableBudget->native_amount = null; | ||||
|         if ($availableBudget->transactionCurrency->id !== $userCurrency->id) { | ||||
|             $converter                      = new ExchangeRateConverter(); | ||||
|   | ||||
| @@ -59,7 +59,7 @@ class BillObserver | ||||
|         if (!Amount::convertToNative($bill->user)) { | ||||
|             return; | ||||
|         } | ||||
|         $userCurrency            = app('amount')->getDefaultCurrencyByUserGroup($bill->user->userGroup); | ||||
|         $userCurrency            = app('amount')->getNativeCurrencyByUserGroup($bill->user->userGroup); | ||||
|         $bill->native_amount_min = null; | ||||
|         $bill->native_amount_max = null; | ||||
|         if ($bill->transactionCurrency->id !== $userCurrency->id) { | ||||
|   | ||||
| @@ -50,7 +50,7 @@ class BudgetLimitObserver | ||||
| 
 | ||||
|             return; | ||||
|         } | ||||
|         $userCurrency               = app('amount')->getDefaultCurrencyByUserGroup($budgetLimit->budget->user->userGroup); | ||||
|         $userCurrency               = app('amount')->getNativeCurrencyByUserGroup($budgetLimit->budget->user->userGroup); | ||||
|         $budgetLimit->native_amount = null; | ||||
|         if ($budgetLimit->transactionCurrency->id !== $userCurrency->id) { | ||||
|             $converter                  = new ExchangeRateConverter(); | ||||
|   | ||||
| @@ -48,7 +48,7 @@ class PiggyBankEventObserver | ||||
|         if (!Amount::convertToNative($event->piggyBank->accounts()->first()->user)) { | ||||
|             return; | ||||
|         } | ||||
|         $userCurrency         = app('amount')->getDefaultCurrencyByUserGroup($event->piggyBank->accounts()->first()->user->userGroup); | ||||
|         $userCurrency         = app('amount')->getNativeCurrencyByUserGroup($event->piggyBank->accounts()->first()->user->userGroup); | ||||
|         $event->native_amount = null; | ||||
|         if ($event->piggyBank->transactionCurrency->id !== $userCurrency->id) { | ||||
|             $converter            = new ExchangeRateConverter(); | ||||
|   | ||||
| @@ -70,7 +70,7 @@ class PiggyBankObserver | ||||
| 
 | ||||
|             return; | ||||
|         } | ||||
|         $userCurrency                    = app('amount')->getDefaultCurrencyByUserGroup($group); | ||||
|         $userCurrency                    = app('amount')->getNativeCurrencyByUserGroup($group); | ||||
|         $piggyBank->native_target_amount = null; | ||||
|         if ($piggyBank->transactionCurrency->id !== $userCurrency->id) { | ||||
|             $converter                       = new ExchangeRateConverter(); | ||||
|   | ||||
| @@ -71,7 +71,7 @@ class TransactionObserver | ||||
|         if (!Amount::convertToNative($transaction->transactionJournal->user)) { | ||||
|             return; | ||||
|         } | ||||
|         $userCurrency                       = app('amount')->getDefaultCurrencyByUserGroup($transaction->transactionJournal->user->userGroup); | ||||
|         $userCurrency                       = app('amount')->getNativeCurrencyByUserGroup($transaction->transactionJournal->user->userGroup); | ||||
|         $transaction->native_amount         = null; | ||||
|         $transaction->native_foreign_amount = null; | ||||
|         // first normal amount
 | ||||
|   | ||||
| @@ -77,7 +77,7 @@ class NetWorth implements NetWorthInterface | ||||
|             return $cache->get(); | ||||
|         } | ||||
|         Log::debug(sprintf('Now in byAccounts("%s", "%s")', $ids, $date->format('Y-m-d H:i:s'))); | ||||
|         $default         = Amount::getDefaultCurrency(); | ||||
|         $default         = Amount::getNativeCurrency(); | ||||
|         $netWorth        = []; | ||||
|         $balances        = Steam::finalAccountsBalance($accounts, $date); | ||||
| 
 | ||||
|   | ||||
| @@ -119,7 +119,7 @@ class EditController extends Controller | ||||
|         } | ||||
|         $request->session()->forget('accounts.edit.fromUpdate'); | ||||
| 
 | ||||
|         $openingBalanceAmount = (string) $repository->getOpeningBalanceAmount($account); | ||||
|         $openingBalanceAmount = (string) $repository->getOpeningBalanceAmount($account, false); | ||||
|         if ('0' === $openingBalanceAmount) { | ||||
|             $openingBalanceAmount = ''; | ||||
|         } | ||||
|   | ||||
| @@ -106,6 +106,7 @@ class IndexController extends Controller | ||||
|                 $account->interestPeriod    = (string) trans(sprintf('firefly.interest_calc_%s', $this->repository->getMetaValue($account, 'interest_period'))); | ||||
|                 $account->accountTypeString = (string) trans(sprintf('firefly.account_type_%s', $account->accountType->type)); | ||||
|                 $account->current_debt      = '0'; | ||||
|                 $account->currency          = $currency ?? $this->defaultCurrency; | ||||
|                 $account->iban              = implode(' ', str_split((string) $account->iban, 4)); | ||||
|             } | ||||
|         ); | ||||
|   | ||||
| @@ -45,7 +45,7 @@ class ConfigurationController extends Controller | ||||
| 
 | ||||
|         $this->middleware( | ||||
|             static function ($request, $next) { | ||||
|                 app('view')->share('title', (string) trans('firefly.administration')); | ||||
|                 app('view')->share('title', (string) trans('firefly.system_settings')); | ||||
|                 app('view')->share('mainTitleIcon', 'fa-hand-spock-o'); | ||||
| 
 | ||||
|                 return $next($request); | ||||
| @@ -73,7 +73,7 @@ class ConfigurationController extends Controller | ||||
|         $siteOwner      = config('firefly.site_owner'); | ||||
| 
 | ||||
|         return view( | ||||
|             'admin.configuration.index', | ||||
|             'settings.configuration.index', | ||||
|             compact('subTitle', 'subTitleIcon', 'singleUserMode', 'isDemoSite', 'siteOwner') | ||||
|         ); | ||||
|     } | ||||
| @@ -96,6 +96,6 @@ class ConfigurationController extends Controller | ||||
|         session()->flash('success', (string) trans('firefly.configuration_updated')); | ||||
|         app('preferences')->mark(); | ||||
| 
 | ||||
|         return redirect()->route('admin.configuration.index'); | ||||
|         return redirect()->route('settings.configuration.index'); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -51,7 +51,7 @@ class HomeController extends Controller | ||||
|     public function index() | ||||
|     { | ||||
|         Log::channel('audit')->info('User visits admin index.'); | ||||
|         $title         = (string) trans('firefly.administration'); | ||||
|         $title         = (string) trans('firefly.system_settings'); | ||||
|         $mainTitleIcon = 'fa-hand-spock-o'; | ||||
|         $email         = auth()->user()->email; | ||||
|         $pref          = app('preferences')->get('remote_guard_alt_email'); | ||||
| @@ -59,6 +59,6 @@ class HomeController extends Controller | ||||
|             $email = $pref->data; | ||||
|         } | ||||
| 
 | ||||
|         return view('admin.index', compact('title', 'mainTitleIcon', 'email')); | ||||
|         return view('settings.index', compact('title', 'mainTitleIcon', 'email')); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -51,7 +51,7 @@ class LinkController extends Controller | ||||
| 
 | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 app('view')->share('title', (string) trans('firefly.administration')); | ||||
|                 app('view')->share('title', (string) trans('firefly.system_settings')); | ||||
|                 app('view')->share('mainTitleIcon', 'fa-hand-spock-o'); | ||||
|                 $this->repository = app(LinkTypeRepositoryInterface::class); | ||||
| 
 | ||||
| @@ -78,7 +78,7 @@ class LinkController extends Controller | ||||
|             $this->rememberPreviousUrl('link-types.create.url'); | ||||
|         } | ||||
| 
 | ||||
|         return view('admin.link.create', compact('subTitle', 'subTitleIcon')); | ||||
|         return view('settings.link.create', compact('subTitle', 'subTitleIcon')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -91,7 +91,7 @@ class LinkController extends Controller | ||||
|         if (false === $linkType->editable) { | ||||
|             $request->session()->flash('error', (string) trans('firefly.cannot_edit_link_type', ['name' => e($linkType->name)])); | ||||
| 
 | ||||
|             return redirect(route('admin.links.index')); | ||||
|             return redirect(route('settings.links.index')); | ||||
|         } | ||||
| 
 | ||||
|         Log::channel('audit')->info(sprintf('User wants to delete link type #%d', $linkType->id)); | ||||
| @@ -111,7 +111,7 @@ class LinkController extends Controller | ||||
|         // put previous url in session
 | ||||
|         $this->rememberPreviousUrl('link-types.delete.url'); | ||||
| 
 | ||||
|         return view('admin.link.delete', compact('linkType', 'subTitle', 'moveTo', 'count')); | ||||
|         return view('settings.link.delete', compact('linkType', 'subTitle', 'moveTo', 'count')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -142,7 +142,7 @@ class LinkController extends Controller | ||||
|         if (false === $linkType->editable) { | ||||
|             $request->session()->flash('error', (string) trans('firefly.cannot_edit_link_type', ['name' => e($linkType->name)])); | ||||
| 
 | ||||
|             return redirect(route('admin.links.index')); | ||||
|             return redirect(route('settings.links.index')); | ||||
|         } | ||||
|         $subTitle     = (string) trans('firefly.edit_link_type', ['name' => $linkType->name]); | ||||
|         $subTitleIcon = 'fa-link'; | ||||
| @@ -155,7 +155,7 @@ class LinkController extends Controller | ||||
|         } | ||||
|         $request->session()->forget('link-types.edit.fromUpdate'); | ||||
| 
 | ||||
|         return view('admin.link.edit', compact('subTitle', 'subTitleIcon', 'linkType')); | ||||
|         return view('settings.link.edit', compact('subTitle', 'subTitleIcon', 'linkType')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -176,7 +176,7 @@ class LinkController extends Controller | ||||
|             } | ||||
|         ); | ||||
| 
 | ||||
|         return view('admin.link.index', compact('subTitle', 'subTitleIcon', 'linkTypes')); | ||||
|         return view('settings.link.index', compact('subTitle', 'subTitleIcon', 'linkTypes')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -192,7 +192,7 @@ class LinkController extends Controller | ||||
| 
 | ||||
|         Log::channel('audit')->info(sprintf('User viewing link type #%d', $linkType->id)); | ||||
| 
 | ||||
|         return view('admin.link.show', compact('subTitle', 'subTitleIcon', 'linkType', 'links')); | ||||
|         return view('settings.link.show', compact('subTitle', 'subTitleIcon', 'linkType', 'links')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -217,7 +217,7 @@ class LinkController extends Controller | ||||
|             // set value so create routine will not overwrite URL:
 | ||||
|             $request->session()->put('link-types.create.fromStore', true); | ||||
| 
 | ||||
|             $redirect = redirect(route('admin.links.create'))->withInput(); | ||||
|             $redirect = redirect(route('settings.links.create'))->withInput(); | ||||
|         } | ||||
| 
 | ||||
|         // redirect to previous URL.
 | ||||
| @@ -234,7 +234,7 @@ class LinkController extends Controller | ||||
|         if (false === $linkType->editable) { | ||||
|             $request->session()->flash('error', (string) trans('firefly.cannot_edit_link_type', ['name' => e($linkType->name)])); | ||||
| 
 | ||||
|             return redirect(route('admin.links.index')); | ||||
|             return redirect(route('settings.links.index')); | ||||
|         } | ||||
| 
 | ||||
|         $data     = [ | ||||
| @@ -253,7 +253,7 @@ class LinkController extends Controller | ||||
|             // set value so edit routine will not overwrite URL:
 | ||||
|             $request->session()->put('link-types.edit.fromUpdate', true); | ||||
| 
 | ||||
|             $redirect = redirect(route('admin.links.edit', [$linkType->id]))->withInput(['return_to_edit' => 1]); | ||||
|             $redirect = redirect(route('settings.links.edit', [$linkType->id]))->withInput(['return_to_edit' => 1]); | ||||
|         } | ||||
| 
 | ||||
|         // redirect to previous URL.
 | ||||
|   | ||||
| @@ -38,7 +38,7 @@ class NotificationController extends Controller | ||||
|     public function index(): View | ||||
|     { | ||||
|         Log::channel('audit')->info('User visits notifications index.'); | ||||
|         $title                          = (string) trans('firefly.administration'); | ||||
|         $title                          = (string) trans('firefly.system_settings'); | ||||
|         $mainTitleIcon                  = 'fa-hand-spock-o'; | ||||
|         $subTitle                       = (string) trans('firefly.title_owner_notifications'); | ||||
|         $subTitleIcon                   = 'envelope-o'; | ||||
| @@ -71,7 +71,7 @@ class NotificationController extends Controller | ||||
|         $forcedAvailability['pushover'] = '' !== $pushoverAppToken && '' !== $pushoverUserToken; | ||||
| 
 | ||||
|         return view( | ||||
|             'admin.notifications.index', | ||||
|             'settings.notifications.index', | ||||
|             compact( | ||||
|                 'title', | ||||
|                 'subTitle', | ||||
| @@ -115,7 +115,7 @@ class NotificationController extends Controller | ||||
| 
 | ||||
|         session()->flash('success', (string) trans('firefly.notification_settings_saved')); | ||||
| 
 | ||||
|         return redirect(route('admin.notification.index')); | ||||
|         return redirect(route('settings.notification.index')); | ||||
|     } | ||||
| 
 | ||||
|     public function testNotification(Request $request): RedirectResponse | ||||
| @@ -140,6 +140,6 @@ class NotificationController extends Controller | ||||
|                 session()->flash('success', (string) trans('firefly.notification_test_executed', ['channel' => $channel])); | ||||
|         } | ||||
| 
 | ||||
|         return redirect(route('admin.notification.index')); | ||||
|         return redirect(route('settings.notification.index')); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -47,7 +47,7 @@ class UpdateController extends Controller | ||||
|         parent::__construct(); | ||||
|         $this->middleware( | ||||
|             static function ($request, $next) { | ||||
|                 app('view')->share('title', (string) trans('firefly.administration')); | ||||
|                 app('view')->share('title', (string) trans('firefly.system_settings')); | ||||
|                 app('view')->share('mainTitleIcon', 'fa-hand-spock-o'); | ||||
| 
 | ||||
|                 return $next($request); | ||||
| @@ -81,7 +81,7 @@ class UpdateController extends Controller | ||||
|             'alpha'  => (string) trans('firefly.update_channel_alpha'), | ||||
|         ]; | ||||
| 
 | ||||
|         return view('admin.update.index', compact('subTitle', 'subTitleIcon', 'selected', 'options', 'channelSelected', 'channelOptions')); | ||||
|         return view('settings.update.index', compact('subTitle', 'subTitleIcon', 'selected', 'options', 'channelSelected', 'channelOptions')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -100,7 +100,7 @@ class UpdateController extends Controller | ||||
|         app('fireflyconfig')->set('update_channel', $channel); | ||||
|         session()->flash('success', (string) trans('firefly.configuration_updated')); | ||||
| 
 | ||||
|         return redirect(route('admin.update-check')); | ||||
|         return redirect(route('settings.update-check')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -112,6 +112,6 @@ class UpdateController extends Controller | ||||
| 
 | ||||
|         session()->flash($release['level'], $release['message']); | ||||
| 
 | ||||
|         return redirect(route('admin.update-check')); | ||||
|         return redirect(route('settings.update-check')); | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -55,7 +55,7 @@ class UserController extends Controller | ||||
| 
 | ||||
|         $this->middleware( | ||||
|             function ($request, $next) { | ||||
|                 app('view')->share('title', (string) trans('firefly.administration')); | ||||
|                 app('view')->share('title', (string) trans('firefly.system_settings')); | ||||
|                 app('view')->share('mainTitleIcon', 'fa-hand-spock-o'); | ||||
|                 $this->repository = app(UserRepositoryInterface::class); | ||||
| 
 | ||||
| @@ -74,12 +74,12 @@ class UserController extends Controller | ||||
|         if ($this->externalIdentity) { | ||||
|             request()->session()->flash('error', trans('firefly.external_user_mgt_disabled')); | ||||
| 
 | ||||
|             return redirect(route('admin.users')); | ||||
|             return redirect(route('settings.users')); | ||||
|         } | ||||
| 
 | ||||
|         $subTitle = (string) trans('firefly.delete_user', ['email' => $user->email]); | ||||
| 
 | ||||
|         return view('admin.users.delete', compact('user', 'subTitle')); | ||||
|         return view('settings.users.delete', compact('user', 'subTitle')); | ||||
|     } | ||||
| 
 | ||||
|     public function deleteInvite(InvitedUser $invitedUser): JsonResponse | ||||
| @@ -108,12 +108,12 @@ class UserController extends Controller | ||||
|         if ($this->externalIdentity) { | ||||
|             request()->session()->flash('error', trans('firefly.external_user_mgt_disabled')); | ||||
| 
 | ||||
|             return redirect(route('admin.users')); | ||||
|             return redirect(route('settings.users')); | ||||
|         } | ||||
|         $this->repository->destroy($user); | ||||
|         session()->flash('success', (string) trans('firefly.user_deleted')); | ||||
| 
 | ||||
|         return redirect(route('admin.users')); | ||||
|         return redirect(route('settings.users')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -144,7 +144,7 @@ class UserController extends Controller | ||||
|             'email_changed' => (string) trans('firefly.block_code_email_changed'), | ||||
|         ]; | ||||
| 
 | ||||
|         return view('admin.users.edit', compact('user', 'canEditDetails', 'subTitle', 'subTitleIcon', 'codes', 'currentUser', 'isAdmin')); | ||||
|         return view('settings.users.edit', compact('user', 'canEditDetails', 'subTitle', 'subTitleIcon', 'codes', 'currentUser', 'isAdmin')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -174,7 +174,7 @@ class UserController extends Controller | ||||
|             } | ||||
|         ); | ||||
| 
 | ||||
|         return view('admin.users.index', compact('subTitle', 'subTitleIcon', 'users', 'allowInvites', 'invitedUsers')); | ||||
|         return view('settings.users.index', compact('subTitle', 'subTitleIcon', 'users', 'allowInvites', 'invitedUsers')); | ||||
|     } | ||||
| 
 | ||||
|     public function invite(InviteUserFormRequest $request): RedirectResponse | ||||
| @@ -186,7 +186,7 @@ class UserController extends Controller | ||||
|         // event!
 | ||||
|         event(new InvitationCreated($invitee)); | ||||
| 
 | ||||
|         return redirect(route('admin.users')); | ||||
|         return redirect(route('settings.users')); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
| @@ -196,14 +196,14 @@ class UserController extends Controller | ||||
|      */ | ||||
|     public function show(User $user) | ||||
|     { | ||||
|         $title         = (string) trans('firefly.administration'); | ||||
|         $title         = (string) trans('firefly.system_settings'); | ||||
|         $mainTitleIcon = 'fa-hand-spock-o'; | ||||
|         $subTitle      = (string) trans('firefly.single_user_administration', ['email' => $user->email]); | ||||
|         $subTitleIcon  = 'fa-user'; | ||||
|         $information   = $this->repository->getUserData($user); | ||||
| 
 | ||||
|         return view( | ||||
|             'admin.users.show', | ||||
|             'settings.users.show', | ||||
|             compact( | ||||
|                 'title', | ||||
|                 'mainTitleIcon', | ||||
| @@ -248,7 +248,7 @@ class UserController extends Controller | ||||
|         if (1 === (int) $request->get('return_to_edit')) { | ||||
|             session()->put('users.edit.fromUpdate', true); | ||||
| 
 | ||||
|             $redirect = redirect(route('admin.users.edit', [$user->id]))->withInput(['return_to_edit' => 1]); | ||||
|             $redirect = redirect(route('settings.users.edit', [$user->id]))->withInput(['return_to_edit' => 1]); | ||||
|         } | ||||
| 
 | ||||
|         // redirect to previous URL.
 | ||||
|   | ||||
| @@ -95,17 +95,22 @@ abstract class Controller extends BaseController | ||||
| 
 | ||||
|         // share is alpha, is beta
 | ||||
|         $isAlpha = false; | ||||
|         $isBeta = false; | ||||
|         $isDevelop = false; | ||||
|         if (str_contains(config('firefly.version'), 'alpha')) { | ||||
|             $isAlpha = true; | ||||
|         } | ||||
|         if (str_contains(config('firefly.version'), 'develop') || str_contains(config('firefly.version'), 'branch')) { | ||||
|             $isDevelop = true; | ||||
|         } | ||||
| 
 | ||||
|         $isBeta = false; | ||||
|         if (str_contains(config('firefly.version'), 'beta')) { | ||||
|             $isBeta = true; | ||||
|         } | ||||
| 
 | ||||
|         View::share('FF_IS_ALPHA', $isAlpha); | ||||
|         View::share('FF_IS_BETA', $isBeta); | ||||
|         View::share('FF_IS_DEVELOP', $isDevelop); | ||||
| 
 | ||||
|         $this->middleware( | ||||
|             function ($request, $next): mixed { | ||||
| @@ -118,7 +123,7 @@ abstract class Controller extends BaseController | ||||
|                 $this->defaultCurrency   =null; | ||||
|                 // get shown-intro-preference:
 | ||||
|                 if (auth()->check()) { | ||||
|                     $this->defaultCurrency   = Amount::getDefaultCurrency(); | ||||
|                     $this->defaultCurrency   = Amount::getNativeCurrency(); | ||||
|                     $language  = Steam::getLanguage(); | ||||
|                     $locale    = Steam::getLocale(); | ||||
|                     $darkMode  = app('preferences')->get('darkMode', 'browser')->data; | ||||
|   | ||||
| @@ -31,6 +31,9 @@ use FireflyIII\Exceptions\FireflyException; | ||||
| use FireflyIII\Http\Middleware\IsDemoUser; | ||||
| use FireflyIII\Models\TransactionType; | ||||
| use FireflyIII\Repositories\PiggyBank\PiggyBankRepositoryInterface; | ||||
| use FireflyIII\Support\Facades\Amount; | ||||
| use FireflyIII\Support\Facades\Preferences; | ||||
| use FireflyIII\Support\Facades\Steam; | ||||
| use FireflyIII\Support\Http\Controllers\GetConfigurationData; | ||||
| use FireflyIII\Support\Models\AccountBalanceCalculator; | ||||
| use FireflyIII\User; | ||||
| @@ -61,7 +64,7 @@ class DebugController extends Controller | ||||
|         $this->middleware(IsDemoUser::class)->except(['displayError']); | ||||
|     } | ||||
| 
 | ||||
|     public function routes(): never | ||||
|     public function routes(Request $request): never | ||||
|     { | ||||
|         if (!auth()->user()->hasRole('owner')) { | ||||
|             throw new NotFoundHttpException(); | ||||
| @@ -69,6 +72,46 @@ class DebugController extends Controller | ||||
| 
 | ||||
|         /** @var iterable $routes */ | ||||
|         $routes = Route::getRoutes(); | ||||
| 
 | ||||
|         if ('true' === $request->get('api')) { | ||||
|             $collection = []; | ||||
|             $i          = 0; | ||||
| 
 | ||||
|             echo 'PATHS="'; | ||||
| 
 | ||||
|             /** @var \Illuminate\Routing\Route $route */ | ||||
|             foreach ($routes as $route) { | ||||
|                 ++$i; | ||||
|                 // skip API and other routes.
 | ||||
|                 if (!str_starts_with($route->uri(), 'api/v1') | ||||
|                 ) { | ||||
|                     continue; | ||||
|                 } | ||||
|                 // skip non GET routes
 | ||||
|                 if (!in_array('GET', $route->methods(), true)) { | ||||
|                     continue; | ||||
|                 } | ||||
|                 // no name route:
 | ||||
|                 if (null === $route->getName()) { | ||||
|                     var_dump($route); | ||||
| 
 | ||||
|                     exit; | ||||
|                 } | ||||
| 
 | ||||
|                 echo substr($route->uri(), 3); | ||||
|                 if (0 === $i % 5) { | ||||
|                     echo '"<br>PATHS="${PATHS},'; | ||||
|                 } | ||||
|                 if (0 !== $i % 5) { | ||||
|                     echo ','; | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|             exit; | ||||
|         } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|         $return = []; | ||||
| 
 | ||||
|         /** @var \Illuminate\Routing\Route $route */ | ||||
| @@ -153,7 +196,7 @@ class DebugController extends Controller | ||||
|      */ | ||||
|     public function flush(Request $request) | ||||
|     { | ||||
|         app('preferences')->mark(); | ||||
|         Preferences::mark(); | ||||
|         $request->session()->forget(['start', 'end', '_previous', 'viewRange', 'range', 'is_custom_range', 'temp-mfa-secret', 'temp-mfa-codes']); | ||||
| 
 | ||||
|         Artisan::call('cache:clear'); | ||||
| @@ -223,8 +266,8 @@ class DebugController extends Controller | ||||
| 
 | ||||
|     private function getSystemInformation(): array | ||||
|     { | ||||
|         $maxFileSize   = app('steam')->phpBytes((string) ini_get('upload_max_filesize')); | ||||
|         $maxPostSize   = app('steam')->phpBytes((string) ini_get('post_max_size')); | ||||
|         $maxFileSize   = Steam::phpBytes((string) ini_get('upload_max_filesize')); | ||||
|         $maxPostSize   = Steam::phpBytes((string) ini_get('post_max_size')); | ||||
|         $drivers       = \DB::availableDrivers(); | ||||
|         $currentDriver = \DB::getDriverName(); | ||||
| 
 | ||||
| @@ -314,7 +357,7 @@ class DebugController extends Controller | ||||
|         ]; | ||||
|     } | ||||
| 
 | ||||
|     private function getuserInfo(): array | ||||
|     private function getUserInfo(): array | ||||
|     { | ||||
|         $userFlags      = $this->getUserFlags(); | ||||
| 
 | ||||
| @@ -324,7 +367,7 @@ class DebugController extends Controller | ||||
|         // set languages, see what happens:
 | ||||
|         $original       = setlocale(LC_ALL, '0'); | ||||
|         $localeAttempts = []; | ||||
|         $parts          = app('steam')->getLocaleArray(app('steam')->getLocale()); | ||||
|         $parts          = Steam::getLocaleArray(Steam::getLocale()); | ||||
|         foreach ($parts as $code) { | ||||
|             $code                  = trim($code); | ||||
|             app('log')->debug(sprintf('Trying to set %s', $code)); | ||||
| @@ -334,14 +377,15 @@ class DebugController extends Controller | ||||
|         setlocale(LC_ALL, (string) $original); | ||||
| 
 | ||||
|         return [ | ||||
|             'user_id'         => auth()->user()->id, | ||||
|             'user_count'      => User::count(), | ||||
|             'user_flags'      => $userFlags, | ||||
|             'user_agent'      => $userAgent, | ||||
|             'locale_attempts' => $localeAttempts, | ||||
|             'locale'          => app('steam')->getLocale(), | ||||
|             'language'        => app('steam')->getLanguage(), | ||||
|             'view_range'      => app('preferences')->get('viewRange', '1M')->data, | ||||
|             'user_id'           => auth()->user()->id, | ||||
|             'user_count'        => User::count(), | ||||
|             'user_flags'        => $userFlags, | ||||
|             'user_agent'        => $userAgent, | ||||
|             'convert_to_native' => Amount::convertToNative(), | ||||
|             'locale_attempts'   => $localeAttempts, | ||||
|             'locale'            => Steam::getLocale(), | ||||
|             'language'          => Steam::getLanguage(), | ||||
|             'view_range'        => Preferences::get('viewRange', '1M')->data, | ||||
|         ]; | ||||
|     } | ||||
| 
 | ||||
|   | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user