mirror of
				https://github.com/firefly-iii/firefly-iii.git
				synced 2025-02-25 18:45:27 -06:00 
			
		
		
		
	
		
			
				
	
	
		
			346 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			346 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * RecurrenceFormRequest.php
 | |
|  * Copyright (c) 2019 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\Http\Requests;
 | |
| 
 | |
| use Carbon\Carbon;
 | |
| use FireflyIII\Exceptions\FireflyException;
 | |
| use FireflyIII\Models\Recurrence;
 | |
| use FireflyIII\Models\TransactionType;
 | |
| use FireflyIII\Rules\ValidRecurrenceRepetitionType;
 | |
| use FireflyIII\Rules\ValidRecurrenceRepetitionValue;
 | |
| use FireflyIII\Validation\AccountValidator;
 | |
| use Illuminate\Validation\Validator;
 | |
| use Log;
 | |
| 
 | |
| /**
 | |
|  * Class RecurrenceFormRequest
 | |
|  */
 | |
| class RecurrenceFormRequest extends Request
 | |
| {
 | |
| 
 | |
|     /**
 | |
|      * Verify the request.
 | |
|      *
 | |
|      * @return bool
 | |
|      */
 | |
|     public function authorize(): bool
 | |
|     {
 | |
|         // Only allow logged in users
 | |
|         return auth()->check();
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Get the data required by the controller.
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     public function getAll(): array
 | |
|     {
 | |
|         $repetitionData = $this->parseRepetitionData();
 | |
|         $return         = [
 | |
|             'recurrence'   => [
 | |
|                 'type'           => $this->string('transaction_type'),
 | |
|                 'title'          => $this->string('title'),
 | |
|                 'description'    => $this->string('recurring_description'),
 | |
|                 'first_date'     => $this->date('first_date'),
 | |
|                 'repeat_until'   => $this->date('repeat_until'),
 | |
|                 'repetitions'    => $this->integer('repetitions'),
 | |
|                 'apply_rules'    => $this->boolean('apply_rules'),
 | |
|                 'active'         => $this->boolean('active'),
 | |
|                 'repetition_end' => $this->string('repetition_end'),
 | |
|             ],
 | |
|             'transactions' => [
 | |
|                 [
 | |
|                     'currency_id'           => $this->integer('transaction_currency_id'),
 | |
|                     'currency_code'         => null,
 | |
|                     'type'                  => $this->string('transaction_type'),
 | |
|                     'description'           => $this->string('transaction_description'),
 | |
|                     'amount'                => $this->string('amount'),
 | |
|                     'foreign_amount'        => null,
 | |
|                     'foreign_currency_id'   => null,
 | |
|                     'foreign_currency_code' => null,
 | |
|                     'budget_id'             => $this->integer('budget_id'),
 | |
|                     'budget_name'           => null,
 | |
|                     'category_id'           => null,
 | |
|                     'category_name'         => $this->string('category'),
 | |
|                     'tags'                  => '' !== $this->string('tags') ? explode(',', $this->string('tags')) : [],
 | |
|                     'piggy_bank_id'         => $this->integer('piggy_bank_id'),
 | |
|                     'piggy_bank_name'       => null,
 | |
|                 ],
 | |
|             ],
 | |
|             'repetitions'  => [
 | |
|                 [
 | |
|                     'type'    => $repetitionData['type'],
 | |
|                     'moment'  => $repetitionData['moment'],
 | |
|                     'skip'    => $this->integer('skip'),
 | |
|                     'weekend' => $this->integer('weekend'),
 | |
|                 ],
 | |
|             ],
 | |
| 
 | |
|         ];
 | |
| 
 | |
|         // fill in foreign currency data
 | |
|         if (null !== $this->float('foreign_amount')) {
 | |
|             $return['transactions'][0]['foreign_amount']      = $this->string('foreign_amount');
 | |
|             $return['transactions'][0]['foreign_currency_id'] = $this->integer('foreign_currency_id');
 | |
|         }
 | |
|         // default values:
 | |
|         $return['transactions'][0]['source_id']        = null;
 | |
|         $return['transactions'][0]['source_name']      = null;
 | |
|         $return['transactions'][0]['destination_id']   = null;
 | |
|         $return['transactions'][0]['destination_name'] = null;
 | |
|         // fill in source and destination account data
 | |
|         switch ($this->string('transaction_type')) {
 | |
|             default:
 | |
|                 throw new FireflyException(sprintf('Cannot handle transaction type "%s"', $this->string('transaction_type'))); // @codeCoverageIgnore
 | |
|             case 'withdrawal':
 | |
|                 $return['transactions'][0]['source_id']      = $this->integer('source_id');
 | |
|                 $return['transactions'][0]['destination_id'] = $this->integer('withdrawal_destination_id');
 | |
|                 break;
 | |
|             case 'deposit':
 | |
|                 $return['transactions'][0]['source_id']      = $this->integer('deposit_source_id');
 | |
|                 $return['transactions'][0]['destination_id'] = $this->integer('destination_id');
 | |
|                 break;
 | |
|             case 'transfer':
 | |
|                 $return['transactions'][0]['source_id']      = $this->integer('source_id');
 | |
|                 $return['transactions'][0]['destination_id'] = $this->integer('destination_id');
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
|         return $return;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * The rules for this request.
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     public function rules(): array
 | |
|     {
 | |
|         $today    = new Carbon;
 | |
|         $tomorrow = Carbon::now()->addDay();
 | |
|         $rules    = [
 | |
|             // mandatory info for recurrence.
 | |
|             'title'                   => 'required|between:1,255|uniqueObjectForUser:recurrences,title',
 | |
|             'first_date'              => 'required|date|after:' . $today->format('Y-m-d'),
 | |
|             'repetition_type'         => ['required', new ValidRecurrenceRepetitionValue, new ValidRecurrenceRepetitionType, 'between:1,20'],
 | |
|             'skip'                    => 'required|numeric|between:0,31',
 | |
| 
 | |
|             // optional for recurrence:
 | |
|             'recurring_description'   => 'between:0,65000',
 | |
|             'active'                  => 'numeric|between:0,1',
 | |
|             'apply_rules'             => 'numeric|between:0,1',
 | |
| 
 | |
|             // mandatory for transaction:
 | |
|             'transaction_description' => 'required|between:1,255',
 | |
|             'transaction_type'        => 'required|in:withdrawal,deposit,transfer',
 | |
|             'transaction_currency_id' => 'required|exists:transaction_currencies,id',
 | |
|             'amount'                  => 'numeric|required|gt:0|max:1000000000',
 | |
|             // mandatory account info:
 | |
|             'source_id'               => 'numeric|belongsToUser:accounts,id|nullable',
 | |
|             'source_name'             => 'between:1,255|nullable',
 | |
|             'destination_id'          => 'numeric|belongsToUser:accounts,id|nullable',
 | |
|             'destination_name'        => 'between:1,255|nullable',
 | |
| 
 | |
|             // foreign amount data:
 | |
|             'foreign_amount'          => 'nullable|gt:0|max:1000000000',
 | |
| 
 | |
|             // optional fields:
 | |
|             'budget_id'               => 'mustExist:budgets,id|belongsToUser:budgets,id|nullable',
 | |
|             'category'                => 'between:1,255|nullable',
 | |
|             'tags'                    => 'between:1,255|nullable',
 | |
|         ];
 | |
|         if ($this->integer('foreign_currency_id') > 0) {
 | |
|             $rules['foreign_currency_id'] = 'exists:transaction_currencies,id';
 | |
|         }
 | |
| 
 | |
|         // if ends after X repetitions, set another rule
 | |
|         if ('times' === $this->string('repetition_end')) {
 | |
|             $rules['repetitions'] = 'required|numeric|between:0,254';
 | |
|         }
 | |
|         // if foreign amount, currency must be  different.
 | |
|         if (null !== $this->float('foreign_amount')) {
 | |
|             $rules['foreign_currency_id'] = 'exists:transaction_currencies,id|different:transaction_currency_id';
 | |
|         }
 | |
| 
 | |
|         // if ends at date X, set another rule.
 | |
|         if ('until_date' === $this->string('repetition_end')) {
 | |
|             $rules['repeat_until'] = 'required|date|after:' . $tomorrow->format('Y-m-d');
 | |
|         }
 | |
| 
 | |
|         // switchc on type to expand rules for source and destination accounts:
 | |
|         switch ($this->string('transaction_type')) {
 | |
|             case strtolower(TransactionType::WITHDRAWAL):
 | |
|                 $rules['source_id']        = 'required|exists:accounts,id|belongsToUser:accounts';
 | |
|                 $rules['destination_name'] = 'between:1,255|nullable';
 | |
|                 break;
 | |
|             case strtolower(TransactionType::DEPOSIT):
 | |
|                 $rules['source_name']    = 'between:1,255|nullable';
 | |
|                 $rules['destination_id'] = 'required|exists:accounts,id|belongsToUser:accounts';
 | |
|                 break;
 | |
|             case strtolower(TransactionType::TRANSFER):
 | |
|                 // this may not work:
 | |
|                 $rules['source_id']      = 'required|exists:accounts,id|belongsToUser:accounts|different:destination_id';
 | |
|                 $rules['destination_id'] = 'required|exists:accounts,id|belongsToUser:accounts|different:source_id';
 | |
| 
 | |
|                 break;
 | |
|             default:
 | |
|                 throw new FireflyException(sprintf('Cannot handle transaction type of type "%s"', $this->string('transaction_type'))); // @codeCoverageIgnore
 | |
|         }
 | |
| 
 | |
|         // update some rules in case the user is editing a post:
 | |
|         /** @var Recurrence $recurrence */
 | |
|         $recurrence = $this->route()->parameter('recurrence');
 | |
|         if ($recurrence instanceof Recurrence) {
 | |
|             $rules['id']         = 'required|numeric|exists:recurrences,id';
 | |
|             $rules['title']      = 'required|between:1,255|uniqueObjectForUser:recurrences,title,' . $recurrence->id;
 | |
|             $rules['first_date'] = 'required|date';
 | |
|         }
 | |
| 
 | |
|         return $rules;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Validates the given account information. Switches on given transaction type.
 | |
|      *
 | |
|      * @param Validator $validator
 | |
|      *
 | |
|      * @throws FireflyException
 | |
|      */
 | |
|     public function validateAccountInformation(Validator $validator): void
 | |
|     {
 | |
|         Log::debug('Now in validateAccountInformation()');
 | |
|         /** @var AccountValidator $accountValidator */
 | |
|         $accountValidator = app(AccountValidator::class);
 | |
|         $data             = $validator->getData();
 | |
|         $transactionType  = $data['transaction_type'] ?? 'invalid';
 | |
| 
 | |
|         $accountValidator->setTransactionType($transactionType);
 | |
| 
 | |
|         // default values:
 | |
|         $sourceId      = null;
 | |
|         $destinationId = null;
 | |
| 
 | |
|         // TODO typeOverrule: the account validator may have another opinion on the transaction type.
 | |
| 
 | |
|         switch ($this->string('transaction_type')) {
 | |
|             default:
 | |
|                 throw new FireflyException(sprintf('Cannot handle transaction type "%s"', $this->string('transaction_type'))); // @codeCoverageIgnore
 | |
|             case 'withdrawal':
 | |
|                 $sourceId      = (int) $data['source_id'];
 | |
|                 $destinationId = (int) $data['withdrawal_destination_id'];
 | |
|                 break;
 | |
|             case 'deposit':
 | |
|                 $sourceId      = (int) $data['deposit_source_id'];
 | |
|                 $destinationId = (int) $data['destination_id'];
 | |
|                 break;
 | |
|             case 'transfer':
 | |
|                 $sourceId      = (int) $data['source_id'];
 | |
|                 $destinationId = (int) $data['destination_id'];
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
| 
 | |
|         // validate source account.
 | |
|         $validSource = $accountValidator->validateSource($sourceId, null, null);
 | |
| 
 | |
|         // do something with result:
 | |
|         if (false === $validSource) {
 | |
|             $message = (string) trans('validation.generic_invalid_source');
 | |
|             $validator->errors()->add('source_id', $message);
 | |
|             $validator->errors()->add('deposit_source_id', $message);
 | |
| 
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         // validate destination account
 | |
|         $validDestination = $accountValidator->validateDestination($destinationId, null, null);
 | |
|         // do something with result:
 | |
|         if (false === $validDestination) {
 | |
|             $message = (string) trans('validation.generic_invalid_destination');
 | |
|             $validator->errors()->add('destination_id', $message);
 | |
|             $validator->errors()->add('withdrawal_destination_id', $message);
 | |
| 
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Configure the validator instance with special rules for after the basic validation rules.
 | |
|      *
 | |
|      * @param Validator $validator
 | |
|      *
 | |
|      * @return void
 | |
|      */
 | |
|     public function withValidator(Validator $validator): void
 | |
|     {
 | |
|         $validator->after(
 | |
|             function (Validator $validator) {
 | |
|                 // validate all account info
 | |
|                 $this->validateAccountInformation($validator);
 | |
|             }
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Parses repetition data.
 | |
|      *
 | |
|      * @return array
 | |
|      */
 | |
|     private function parseRepetitionData(): array
 | |
|     {
 | |
|         $value  = $this->string('repetition_type');
 | |
|         $return = [
 | |
|             'type'   => '',
 | |
|             'moment' => '',
 | |
|         ];
 | |
| 
 | |
|         if ('daily' === $value) {
 | |
|             $return['type'] = $value;
 | |
|         }
 | |
|         //monthly,17
 | |
|         //ndom,3,7
 | |
|         if (in_array(substr($value, 0, 6), ['yearly', 'weekly'])) {
 | |
|             $return['type']   = substr($value, 0, 6);
 | |
|             $return['moment'] = substr($value, 7);
 | |
|         }
 | |
|         if (0 === strpos($value, 'monthly')) {
 | |
|             $return['type']   = substr($value, 0, 7);
 | |
|             $return['moment'] = substr($value, 8);
 | |
|         }
 | |
|         if (0 === strpos($value, 'ndom')) {
 | |
|             $return['type']   = substr($value, 0, 4);
 | |
|             $return['moment'] = substr($value, 5);
 | |
|         }
 | |
| 
 | |
|         return $return;
 | |
| 
 | |
| 
 | |
|     }
 | |
| }
 |