| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | <?php | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * SelectController.php | 
					
						
							| 
									
										
										
										
											2020-01-31 07:32:04 +01:00
										 |  |  |  * Copyright (c) 2019 james@firefly-iii.org | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This file is part of Firefly III (https://github.com/firefly-iii). | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * 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. | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * GNU Affero General Public License for more details. | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2019-10-02 06:37:26 +02:00
										 |  |  |  * 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/>. | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | declare(strict_types=1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FireflyIII\Http\Controllers\Rule; | 
					
						
							| 
									
										
										
										
											2021-03-28 11:46:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | use Carbon\Carbon; | 
					
						
							| 
									
										
										
										
											2022-11-02 06:25:37 +01:00
										 |  |  | use FireflyIII\Exceptions\FireflyException; | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | use FireflyIII\Http\Controllers\Controller; | 
					
						
							|  |  |  | use FireflyIII\Http\Requests\SelectTransactionsRequest; | 
					
						
							|  |  |  | use FireflyIII\Http\Requests\TestRuleFormRequest; | 
					
						
							|  |  |  | use FireflyIII\Models\Rule; | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  | use FireflyIII\Models\RuleTrigger; | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | use FireflyIII\Support\Http\Controllers\RuleManagement; | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  | use FireflyIII\TransactionRules\Engine\RuleEngineInterface; | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | use FireflyIII\User; | 
					
						
							| 
									
										
										
										
											2020-03-17 15:01:00 +01:00
										 |  |  | use Illuminate\Contracts\View\Factory; | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | use Illuminate\Http\JsonResponse; | 
					
						
							|  |  |  | use Illuminate\Http\RedirectResponse; | 
					
						
							|  |  |  | use Illuminate\Support\Collection; | 
					
						
							| 
									
										
										
										
											2023-04-02 19:42:06 +02:00
										 |  |  | use Illuminate\View\View; | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							| 
									
										
										
										
											2018-07-22 08:10:16 +02:00
										 |  |  |  * Class SelectController. | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | class SelectController extends Controller | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-04-06 17:00:16 +02:00
										 |  |  |     use RuleManagement; | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * RuleController constructor. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function __construct() | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         parent::__construct(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->middleware( | 
					
						
							| 
									
										
										
										
											2023-11-04 14:18:49 +01:00
										 |  |  |             static function ($request, $next) { | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |                 app('view')->share('title', (string)trans('firefly.rules')); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |                 app('view')->share('mainTitleIcon', 'fa-random'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 return $next($request); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Execute the given rule on a set of existing transactions. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function execute(SelectTransactionsRequest $request, Rule $rule): RedirectResponse | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // Get parameters specified by the user
 | 
					
						
							|  |  |  |         /** @var User $user */ | 
					
						
							|  |  |  |         $user      = auth()->user(); | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         $accounts  = implode(',', $request->get('accounts')); | 
					
						
							|  |  |  |         $startDate = new Carbon($request->get('start')); | 
					
						
							|  |  |  |         $endDate   = new Carbon($request->get('end')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // create new rule engine:
 | 
					
						
							|  |  |  |         $newRuleEngine = app(RuleEngineInterface::class); | 
					
						
							|  |  |  |         $newRuleEngine->setUser($user); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // add extra operators:
 | 
					
						
							|  |  |  |         $newRuleEngine->addOperator(['type' => 'date_after', 'value' => $startDate->format('Y-m-d')]); | 
					
						
							|  |  |  |         $newRuleEngine->addOperator(['type' => 'date_before', 'value' => $endDate->format('Y-m-d')]); | 
					
						
							|  |  |  |         $newRuleEngine->addOperator(['type' => 'account_id', 'value' => $accounts]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // set rules:
 | 
					
						
							|  |  |  |         $newRuleEngine->setRules(new Collection([$rule])); | 
					
						
							|  |  |  |         $newRuleEngine->fire(); | 
					
						
							| 
									
										
										
										
											2021-02-08 15:12:04 +01:00
										 |  |  |         $resultCount = $newRuleEngine->getResults(); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-05 08:15:17 +01:00
										 |  |  |         session()->flash('success', trans_choice('firefly.applied_rule_selection', $resultCount, ['title' => $rule->title])); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return redirect()->route('rules.index'); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-03-28 11:46:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |     /** | 
					
						
							| 
									
										
										
										
											2018-07-22 08:10:16 +02:00
										 |  |  |      * View to select transactions by a rule. | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  |     public function selectTransactions(Rule $rule): Factory|RedirectResponse|View | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         if (false === $rule->active) { | 
					
						
							|  |  |  |             session()->flash('warning', trans('firefly.cannot_fire_inactive_rules')); | 
					
						
							| 
									
										
										
										
											2021-02-08 15:12:04 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-13 13:48:52 +02:00
										 |  |  |             return redirect(route('rules.index')); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |         // does the user have shared accounts?
 | 
					
						
							| 
									
										
										
										
											2023-02-11 07:36:45 +01:00
										 |  |  |         $first    = session('first', today(config('app.timezone'))->subYear())->format('Y-m-d'); | 
					
						
							|  |  |  |         $today    = today(config('app.timezone'))->format('Y-m-d'); | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |         $subTitle = (string)trans('firefly.apply_rule_selection', ['title' => $rule->title]); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-29 14:11:12 +01:00
										 |  |  |         return view('rules.rule.select-transactions', compact('first', 'today', 'rule', 'subTitle')); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * This method allows the user to test a certain set of rule triggers. The rule triggers are passed along | 
					
						
							|  |  |  |      * using the URL parameters (GET), and are usually put there using a Javascript thing. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2023-02-22 18:03:31 +01:00
										 |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function testTriggers(TestRuleFormRequest $request): JsonResponse | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         // build fake rule
 | 
					
						
							| 
									
										
										
										
											2023-12-10 06:51:59 +01:00
										 |  |  |         $rule = new Rule(); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-28 04:45:07 +01:00
										 |  |  |         /** @var \Illuminate\Database\Eloquent\Collection<int, RuleTrigger> $triggers */ | 
					
						
							| 
									
										
										
										
											2022-10-30 14:24:28 +01:00
										 |  |  |         $triggers     = new Collection(); | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         $rule->strict = '1' === $request->get('strict'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |         // build trigger array from response
 | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         $textTriggers = $this->getValidTriggerList($request); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         // warn if nothing.
 | 
					
						
							| 
									
										
										
										
											2022-11-04 05:11:05 +01:00
										 |  |  |         if (0 === count($textTriggers)) { | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |             return response()->json(['html' => '', 'warning' => (string)trans('firefly.warning_no_valid_triggers')]); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         foreach ($textTriggers as $textTrigger) { | 
					
						
							| 
									
										
										
										
											2023-04-02 19:42:06 +02:00
										 |  |  |             $trigger                  = new RuleTrigger(); | 
					
						
							|  |  |  |             $trigger->trigger_type    = $textTrigger['type']; | 
					
						
							|  |  |  |             $trigger->trigger_value   = $textTrigger['value']; | 
					
						
							|  |  |  |             $trigger->stop_processing = $textTrigger['stop_processing']; | 
					
						
							|  |  |  |             if ($textTrigger['prohibited']) { | 
					
						
							|  |  |  |                 $trigger->trigger_type = sprintf('-%s', $textTrigger['type']); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |             $triggers->push($trigger); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         $rule->ruleTriggers = $triggers; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // create new rule engine:
 | 
					
						
							| 
									
										
										
										
											2021-05-01 18:54:11 +02:00
										 |  |  |         /** @var RuleEngineInterface $newRuleEngine */ | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         $newRuleEngine = app(RuleEngineInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // set rules:
 | 
					
						
							|  |  |  |         $newRuleEngine->setRules(new Collection([$rule])); | 
					
						
							| 
									
										
										
										
											2023-03-14 18:09:44 +01:00
										 |  |  |         $newRuleEngine->setRefreshTriggers(false); | 
					
						
							| 
									
										
										
										
											2020-08-24 07:31:50 +02:00
										 |  |  |         $collection = $newRuleEngine->find(); | 
					
						
							| 
									
										
										
										
											2020-08-24 18:31:10 +02:00
										 |  |  |         $collection = $collection->slice(0, 20); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Warn the user if only a subset of transactions is returned
 | 
					
						
							|  |  |  |         $warning = ''; | 
					
						
							| 
									
										
										
										
											2022-11-04 05:11:05 +01:00
										 |  |  |         if (0 === count($collection)) { | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |             $warning = (string)trans('firefly.warning_no_matching_transactions'); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Return json response
 | 
					
						
							|  |  |  |         $view = 'ERROR, see logs.'; | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |         try { | 
					
						
							| 
									
										
										
										
											2022-01-29 14:11:12 +01:00
										 |  |  |             $view = view('list.journals-array-tiny', ['groups' => $collection])->render(); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  |         } catch (\Throwable $exception) { | 
					
						
							| 
									
										
										
										
											2023-10-29 06:32:00 +01:00
										 |  |  |             app('log')->error(sprintf('Could not render view in testTriggers(): %s', $exception->getMessage())); | 
					
						
							|  |  |  |             app('log')->error($exception->getTraceAsString()); | 
					
						
							| 
									
										
										
										
											2019-07-01 20:22:35 +02:00
										 |  |  |             $view = sprintf('Could not render list.journals-tiny: %s', $exception->getMessage()); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 06:25:37 +01:00
										 |  |  |             throw new FireflyException($view, 0, $exception); | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-18 10:26:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 14:34:56 +02:00
										 |  |  |         return response()->json(['html' => $view, 'warning' => $warning]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-27 07:12:44 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * This method allows the user to test a certain set of rule triggers. The rule triggers are grabbed from | 
					
						
							|  |  |  |      * the rule itself. | 
					
						
							|  |  |  |      * | 
					
						
							| 
									
										
										
										
											2023-02-22 18:03:31 +01:00
										 |  |  |      * @throws FireflyException | 
					
						
							| 
									
										
										
										
											2020-08-27 07:12:44 +02:00
										 |  |  |      */ | 
					
						
							|  |  |  |     public function testTriggersByRule(Rule $rule): JsonResponse | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $triggers = $rule->ruleTriggers; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-04 05:11:05 +01:00
										 |  |  |         if (0 === count($triggers)) { | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |             return response()->json(['html' => '', 'warning' => (string)trans('firefly.warning_no_valid_triggers')]); | 
					
						
							| 
									
										
										
										
											2020-08-27 07:12:44 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         // create new rule engine:
 | 
					
						
							|  |  |  |         $newRuleEngine = app(RuleEngineInterface::class); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // set rules:
 | 
					
						
							|  |  |  |         $newRuleEngine->setRules(new Collection([$rule])); | 
					
						
							|  |  |  |         $collection = $newRuleEngine->find(); | 
					
						
							|  |  |  |         $collection = $collection->slice(0, 20); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $warning = ''; | 
					
						
							| 
									
										
										
										
											2022-11-04 05:11:05 +01:00
										 |  |  |         if (0 === count($collection)) { | 
					
						
							| 
									
										
										
										
											2022-12-29 19:42:26 +01:00
										 |  |  |             $warning = (string)trans('firefly.warning_no_matching_transactions'); | 
					
						
							| 
									
										
										
										
											2020-08-27 07:12:44 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Return json response
 | 
					
						
							|  |  |  |         $view = 'ERROR, see logs.'; | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-27 07:12:44 +02:00
										 |  |  |         try { | 
					
						
							| 
									
										
										
										
											2022-01-29 14:11:12 +01:00
										 |  |  |             $view = view('list.journals-array-tiny', ['groups' => $collection])->render(); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  |         } catch (\Throwable $exception) { | 
					
						
							| 
									
										
										
										
											2022-11-02 06:25:37 +01:00
										 |  |  |             $message = sprintf('Could not render view in testTriggersByRule(): %s', $exception->getMessage()); | 
					
						
							| 
									
										
										
										
											2023-10-29 06:32:00 +01:00
										 |  |  |             app('log')->error($message); | 
					
						
							|  |  |  |             app('log')->error($exception->getTraceAsString()); | 
					
						
							| 
									
										
										
										
											2023-12-20 19:35:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-02 06:25:37 +01:00
										 |  |  |             throw new FireflyException($message, 0, $exception); | 
					
						
							| 
									
										
										
										
											2020-08-27 07:12:44 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-18 10:26:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-27 07:12:44 +02:00
										 |  |  |         return response()->json(['html' => $view, 'warning' => $warning]); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-22 20:32:02 +02:00
										 |  |  | } |