mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2025-01-18 04:23:03 -06:00
347 lines
14 KiB
PHP
347 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\Factory\CategoryFactory;
|
|
use FireflyIII\Models\Recurrence;
|
|
use FireflyIII\Models\TransactionType;
|
|
use FireflyIII\Rules\ValidRecurrenceRepetitionType;
|
|
use FireflyIII\Rules\ValidRecurrenceRepetitionValue;
|
|
use FireflyIII\Support\Request\ChecksLogin;
|
|
use FireflyIII\Support\Request\ConvertsDataTypes;
|
|
use FireflyIII\Validation\AccountValidator;
|
|
use Illuminate\Foundation\Http\FormRequest;
|
|
use Illuminate\Validation\Validator;
|
|
use Log;
|
|
|
|
/**
|
|
* Class RecurrenceFormRequest
|
|
*/
|
|
class RecurrenceFormRequest extends FormRequest
|
|
{
|
|
use ConvertsDataTypes, ChecksLogin;
|
|
|
|
/**
|
|
* Get the data required by the controller.
|
|
*
|
|
* @return array
|
|
* @throws FireflyException
|
|
*
|
|
*/
|
|
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'),
|
|
'nr_of_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;
|
|
}
|
|
|
|
// replace category name with a new category:
|
|
$factory = app(CategoryFactory::class);
|
|
$factory->setUser(auth()->user());
|
|
foreach ($return['transactions'] as $index => $transaction) {
|
|
$categoryName = $transaction['category_name'] ?? null;
|
|
if (null !== $categoryName) {
|
|
$category = $factory->findOrCreate(null, $categoryName);
|
|
if (null !== $category) {
|
|
$return['transactions'][$index]['category_id'] = $category->id;
|
|
}
|
|
}
|
|
}
|
|
|
|
return $return;
|
|
}
|
|
|
|
/**
|
|
* 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;
|
|
}
|
|
|
|
/**
|
|
* The rules for this request.
|
|
*
|
|
* @return array
|
|
* @throws FireflyException
|
|
*
|
|
*/
|
|
public function rules(): array
|
|
{
|
|
$today = today(config('app.timezone'));
|
|
$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;
|
|
}
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
);
|
|
}
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
}
|
|
}
|