Files
firefly-iii/app/Api/V1/Controllers/TransactionController.php

407 lines
15 KiB
PHP
Raw Normal View History

2018-02-13 21:04:15 +01:00
<?php
2018-05-11 10:08:34 +02:00
2018-02-13 21:04:15 +01:00
/**
* TransactionController.php
* Copyright (c) 2019 james@firefly-iii.org
2018-02-13 21:04:15 +01:00
*
* This file is part of Firefly III (https://github.com/firefly-iii).
2018-02-13 21:04:15 +01: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-02-13 21:04:15 +01:00
*
* This program is distributed in the hope that it will be useful,
2018-02-13 21:04:15 +01:00
* 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.
2018-02-13 21:04:15 +01: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-02-13 21:04:15 +01:00
*/
2018-05-11 10:08:34 +02:00
declare(strict_types=1);
2018-02-13 21:04:15 +01:00
namespace FireflyIII\Api\V1\Controllers;
2018-02-16 16:42:13 +01:00
use FireflyIII\Api\V1\Requests\TransactionStoreRequest;
use FireflyIII\Api\V1\Requests\TransactionUpdateRequest;
2019-03-30 07:09:52 +01:00
use FireflyIII\Events\StoredTransactionGroup;
use FireflyIII\Events\UpdatedTransactionGroup;
use FireflyIII\Exceptions\DuplicateTransactionException;
2020-03-12 05:06:42 +01:00
use FireflyIII\Exceptions\FireflyException;
2019-03-30 07:09:52 +01:00
use FireflyIII\Helpers\Collector\GroupCollectorInterface;
use FireflyIII\Models\TransactionGroup;
use FireflyIII\Models\TransactionJournal;
use FireflyIII\Repositories\Journal\JournalAPIRepositoryInterface;
2018-02-13 21:04:15 +01:00
use FireflyIII\Repositories\Journal\JournalRepositoryInterface;
use FireflyIII\Repositories\TransactionGroup\TransactionGroupRepositoryInterface;
2018-12-08 21:26:20 +01:00
use FireflyIII\Support\Http\Api\TransactionFilter;
2018-12-04 19:36:54 +01:00
use FireflyIII\Transformers\AttachmentTransformer;
use FireflyIII\Transformers\PiggyBankEventTransformer;
2019-03-30 07:09:52 +01:00
use FireflyIII\Transformers\TransactionGroupTransformer;
2020-07-31 06:19:48 +02:00
use FireflyIII\Transformers\TransactionLinkTransformer;
2018-07-05 06:10:35 +02:00
use FireflyIII\User;
use Illuminate\Http\JsonResponse;
2018-02-13 21:04:15 +01:00
use Illuminate\Http\Request;
2019-09-26 20:57:24 +02:00
use Illuminate\Support\Collection;
2018-02-16 16:42:13 +01:00
use League\Fractal\Pagination\IlluminatePaginatorAdapter;
use League\Fractal\Resource\Collection as FractalCollection;
2019-03-30 07:09:52 +01:00
use League\Fractal\Resource\Item;
use Log;
2019-03-30 07:09:52 +01:00
use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
2018-02-13 21:04:15 +01:00
/**
* Class TransactionController
*/
class TransactionController extends Controller
{
2018-12-08 21:26:20 +01:00
use TransactionFilter;
2018-02-13 21:04:15 +01:00
2020-07-31 06:19:48 +02:00
private TransactionGroupRepositoryInterface $groupRepository;
2020-07-31 09:42:00 +02:00
2020-07-31 06:19:48 +02:00
private JournalAPIRepositoryInterface $journalAPIRepository;
2020-07-31 09:42:00 +02:00
2020-07-31 06:19:48 +02:00
private JournalRepositoryInterface $repository;
2020-07-31 09:24:08 +02:00
2018-02-13 21:04:15 +01:00
/**
* TransactionController constructor.
*
* @codeCoverageIgnore
2018-02-13 21:04:15 +01:00
*/
public function __construct()
{
parent::__construct();
$this->middleware(
function ($request, $next) {
2018-07-05 06:10:35 +02:00
/** @var User $admin */
$admin = auth()->user();
$this->repository = app(JournalRepositoryInterface::class);
$this->groupRepository = app(TransactionGroupRepositoryInterface::class);
$this->journalAPIRepository = app(JournalAPIRepositoryInterface::class);
2018-07-05 06:10:35 +02:00
$this->repository->setUser($admin);
$this->groupRepository->setUser($admin);
$this->journalAPIRepository->setUser($admin);
2018-02-13 21:04:15 +01:00
return $next($request);
}
);
}
2018-12-04 19:36:54 +01:00
/**
2019-09-26 20:57:24 +02:00
* @param TransactionGroup $transactionGroup
2018-12-04 19:36:54 +01:00
*
* @return JsonResponse
* @codeCoverageIgnore
2018-12-04 19:36:54 +01:00
*/
2019-09-26 20:57:24 +02:00
public function attachments(TransactionGroup $transactionGroup): JsonResponse
2018-12-04 19:36:54 +01:00
{
2019-09-04 17:39:39 +02:00
$manager = $this->getManager();
2019-09-26 20:57:24 +02:00
$attachments = new Collection;
foreach ($transactionGroup->transactionJournals as $transactionJournal) {
$attachments = $this->journalAPIRepository->getAttachments($transactionJournal)->merge($attachments);
}
2018-12-15 22:03:05 +01:00
/** @var AttachmentTransformer $transformer */
$transformer = app(AttachmentTransformer::class);
$transformer->setParameters($this->parameters);
$resource = new FractalCollection($attachments, $transformer, 'attachments');
2018-12-04 19:36:54 +01:00
return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json');
}
2020-07-31 06:19:48 +02:00
/**
* @param TransactionGroup $transactionGroup
*
* @return JsonResponse
* @codeCoverageIgnore
*/
public function transactionLinks(TransactionGroup $transactionGroup): JsonResponse
{
$manager = $this->getManager();
$journalLinks = new Collection;
foreach ($transactionGroup->transactionJournals as $transactionJournal) {
$journalLinks = $this->journalAPIRepository->getJournalLinks($transactionJournal)->merge($journalLinks);
}
/** @var TransactionLinkTransformer $transformer */
$transformer = app(TransactionLinkTransformer::class);
$transformer->setParameters($this->parameters);
$resource = new FractalCollection($journalLinks, $transformer, 'transaction_links');
return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json');
}
2018-02-13 21:04:15 +01:00
/**
* Remove the specified resource from storage.
*
2019-03-30 07:09:52 +01:00
* @param TransactionGroup $transactionGroup
2018-02-13 21:04:15 +01:00
*
2018-07-05 06:10:35 +02:00
* @return JsonResponse
* @codeCoverageIgnore
2018-02-13 21:04:15 +01:00
*/
2019-03-30 07:09:52 +01:00
public function delete(TransactionGroup $transactionGroup): JsonResponse
2018-02-13 21:04:15 +01:00
{
2019-03-30 07:09:52 +01:00
$this->repository->destroyGroup($transactionGroup);
return response()->json([], 204);
}
/**
* Remove the specified resource from storage.
*
* @param TransactionJournal $transactionJournal
*
* @codeCoverageIgnore
2019-03-30 07:09:52 +01:00
* @return JsonResponse
*/
public function deleteJournal(TransactionJournal $transactionJournal): JsonResponse
{
$this->repository->destroyJournal($transactionJournal);
2018-02-13 21:04:15 +01:00
return response()->json([], 204);
}
/**
* Show all transactions.
*
2018-02-13 21:04:15 +01:00
* @param Request $request
*
2018-07-05 06:10:35 +02:00
* @return JsonResponse
* @codeCoverageIgnore
2018-02-13 21:04:15 +01:00
*/
2018-07-05 06:10:35 +02:00
public function index(Request $request): JsonResponse
2018-02-13 21:04:15 +01:00
{
$pageSize = (int) app('preferences')->getForUser(auth()->user(), 'listPageSize', 50)->data;
2018-07-05 18:02:02 +02:00
$type = $request->get('type') ?? 'default';
2018-02-16 16:42:13 +01:00
$this->parameters->set('type', $type);
2018-12-08 21:26:20 +01:00
$types = $this->mapTransactionTypes($this->parameters->get('type'));
2019-09-04 17:39:39 +02:00
$manager = $this->getManager();
2018-07-05 06:10:35 +02:00
/** @var User $admin */
$admin = auth()->user();
2019-03-30 07:09:52 +01:00
// use new group collector:
/** @var GroupCollectorInterface $collector */
$collector = app(GroupCollectorInterface::class);
$collector
->setUser($admin)
// all info needed for the API:
->withAPIInformation()
// set page size:
->setLimit($pageSize)
// set page to retrieve
->setPage($this->parameters->get('page'))
// set types of transactions to return.
->setTypes($types);
2018-02-16 16:42:13 +01:00
2018-04-02 14:17:11 +02:00
if (null !== $this->parameters->get('start') && null !== $this->parameters->get('end')) {
2018-02-16 16:42:13 +01:00
$collector->setRange($this->parameters->get('start'), $this->parameters->get('end'));
}
2019-03-30 07:09:52 +01:00
$paginator = $collector->getPaginatedGroups();
2018-02-16 16:42:13 +01:00
$paginator->setPath(route('api.v1.transactions.index') . $this->buildParams());
$transactions = $paginator->getCollection();
2019-03-30 07:09:52 +01:00
/** @var TransactionGroupTransformer $transformer */
$transformer = app(TransactionGroupTransformer::class);
2018-12-15 22:03:05 +01:00
$transformer->setParameters($this->parameters);
$resource = new FractalCollection($transactions, $transformer, 'transactions');
2018-02-13 21:04:15 +01:00
$resource->setPaginator(new IlluminatePaginatorAdapter($paginator));
2018-02-18 10:31:15 +01:00
return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json');
2018-02-13 21:04:15 +01:00
}
2018-12-12 20:30:25 +01:00
/**
2019-09-26 20:57:24 +02:00
* @param TransactionGroup $transactionGroup
2018-12-12 20:30:25 +01:00
*
* @return JsonResponse
* @codeCoverageIgnore
2018-12-12 20:30:25 +01:00
*/
2019-09-26 20:57:24 +02:00
public function piggyBankEvents(TransactionGroup $transactionGroup): JsonResponse
2018-12-12 20:30:25 +01:00
{
2019-09-04 17:39:39 +02:00
$manager = $this->getManager();
2019-09-26 20:57:24 +02:00
$events = new Collection;
foreach ($transactionGroup->transactionJournals as $transactionJournal) {
$events = $this->journalAPIRepository->getPiggyBankEvents($transactionJournal)->merge($events);
}
2018-12-15 22:03:05 +01:00
/** @var PiggyBankEventTransformer $transformer */
$transformer = app(PiggyBankEventTransformer::class);
$transformer->setParameters($this->parameters);
$resource = new FractalCollection($events, $transformer, 'piggy_bank_events');
2018-12-12 20:30:25 +01:00
return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json');
}
2018-02-13 21:04:15 +01:00
/**
* Show a single transaction.
*
2019-03-30 07:09:52 +01:00
* @param TransactionGroup $transactionGroup
2018-02-13 21:04:15 +01:00
*
2018-07-05 06:10:35 +02:00
* @return JsonResponse
* @codeCoverageIgnore
2018-02-13 21:04:15 +01:00
*/
2019-09-04 17:39:39 +02:00
public function show(TransactionGroup $transactionGroup): JsonResponse
2018-02-13 21:04:15 +01:00
{
2019-09-04 17:39:39 +02:00
$manager = $this->getManager();
2019-03-30 07:09:52 +01:00
/** @var User $admin */
$admin = auth()->user();
// use new group collector:
/** @var GroupCollectorInterface $collector */
$collector = app(GroupCollectorInterface::class);
$collector
->setUser($admin)
// filter on transaction group.
->setTransactionGroup($transactionGroup)
// all info needed for the API:
->withAPIInformation();
$selectedGroup = $collector->getGroups()->first();
if (null === $selectedGroup) {
throw new NotFoundHttpException();
2018-02-16 16:42:13 +01:00
}
2019-03-30 07:09:52 +01:00
/** @var TransactionGroupTransformer $transformer */
$transformer = app(TransactionGroupTransformer::class);
2018-12-15 22:03:05 +01:00
$transformer->setParameters($this->parameters);
2019-03-30 07:09:52 +01:00
$resource = new Item($selectedGroup, $transformer, 'transactions');
2018-02-16 16:42:13 +01:00
2018-02-18 10:31:15 +01:00
return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json');
2018-02-13 21:04:15 +01:00
}
2019-10-21 18:32:13 +02:00
/**
* Show a single transaction, by transaction journal.
*
* @param TransactionJournal $transactionJournal
*
* @return JsonResponse
* @codeCoverageIgnore
*/
public function showByJournal(TransactionJournal $transactionJournal): JsonResponse
{
return $this->show($transactionJournal->transactionGroup);
}
2018-02-13 21:04:15 +01:00
/**
* Store a new transaction.
*
* @param TransactionStoreRequest $request
2018-03-11 16:24:07 +01:00
*
* @throws FireflyException
2018-07-05 06:10:35 +02:00
* @return JsonResponse
2018-02-13 21:04:15 +01:00
*/
public function store(TransactionStoreRequest $request): JsonResponse
2018-02-13 21:04:15 +01:00
{
2020-02-22 11:05:16 +01:00
Log::debug('Now in API TransactionController::store()');
2019-05-24 05:28:41 +02:00
$data = $request->getAll();
$data['user'] = auth()->user()->id;
Log::channel('audit')
->info('Store new transaction over API.', $data);
try {
$transactionGroup = $this->groupRepository->store($data);
} catch (DuplicateTransactionException $e) {
2020-05-16 06:59:15 +02:00
Log::warning('Caught a duplicate transaction. Return error message.');
// return bad validation message.
// TODO use Laravel's internal validation thing to do this.
$response = [
'message' => 'The given data was invalid.',
'errors' => [
'transactions.0.description' => [$e->getMessage()],
],
];
2020-03-12 05:06:42 +01:00
return response()->json($response, 422);
2020-03-15 08:16:16 +01:00
} catch (FireflyException $e) {
2020-03-12 05:06:42 +01:00
Log::warning('Caught an exception. Return error message.');
Log::error($e->getMessage());
// return bad validation message.
// TODO use Laravel's internal validation thing to do this.
$response = [
'message' => 'The given data was invalid.',
'errors' => [
'transactions.0.description' => [sprintf('Internal exception: %s', $e->getMessage())],
2020-03-12 05:06:42 +01:00
],
];
return response()->json($response, 422);
}
2020-02-21 20:47:10 +01:00
app('preferences')->mark();
event(new StoredTransactionGroup($transactionGroup, $data['apply_rules'] ?? true));
2018-09-26 20:34:24 +02:00
2019-09-04 17:39:39 +02:00
$manager = $this->getManager();
2019-03-30 07:09:52 +01:00
/** @var User $admin */
$admin = auth()->user();
// use new group collector:
/** @var GroupCollectorInterface $collector */
$collector = app(GroupCollectorInterface::class);
$collector
->setUser($admin)
// filter on transaction group.
->setTransactionGroup($transactionGroup)
// all info needed for the API:
->withAPIInformation();
$selectedGroup = $collector->getGroups()->first();
if (null === $selectedGroup) {
throw new FireflyException('Cannot find transaction. Possibly, a rule deleted this transaction after its creation.');
2018-02-16 16:57:27 +01:00
}
2019-03-30 07:09:52 +01:00
/** @var TransactionGroupTransformer $transformer */
$transformer = app(TransactionGroupTransformer::class);
2018-12-15 22:03:05 +01:00
$transformer->setParameters($this->parameters);
2019-03-30 07:09:52 +01:00
$resource = new Item($selectedGroup, $transformer, 'transactions');
2018-02-16 16:57:27 +01:00
2018-02-18 10:31:15 +01:00
return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json');
2018-02-13 21:04:15 +01:00
}
/**
* Update a transaction.
*
* @param TransactionUpdateRequest $request
2019-09-04 17:39:39 +02:00
* @param TransactionGroup $transactionGroup
2018-02-13 21:04:15 +01:00
*
2018-07-05 06:10:35 +02:00
* @return JsonResponse
2018-02-13 21:04:15 +01:00
*/
public function update(TransactionUpdateRequest $request, TransactionGroup $transactionGroup): JsonResponse
2018-02-13 21:04:15 +01:00
{
Log::debug('Now in update routine.');
2019-03-30 07:09:52 +01:00
$data = $request->getAll();
$transactionGroup = $this->groupRepository->update($transactionGroup, $data);
2019-09-04 17:39:39 +02:00
$manager = $this->getManager();
2018-02-13 21:04:15 +01:00
2020-02-21 20:47:10 +01:00
app('preferences')->mark();
event(new UpdatedTransactionGroup($transactionGroup, $data['apply_rules'] ?? true));
2019-03-30 07:09:52 +01:00
/** @var User $admin */
$admin = auth()->user();
// use new group collector:
/** @var GroupCollectorInterface $collector */
$collector = app(GroupCollectorInterface::class);
$collector
->setUser($admin)
// filter on transaction group.
->setTransactionGroup($transactionGroup)
// all info needed for the API:
->withAPIInformation();
$selectedGroup = $collector->getGroups()->first();
if (null === $selectedGroup) {
2019-04-16 16:20:46 +02:00
throw new NotFoundHttpException(); // @codeCoverageIgnore
}
2019-03-30 07:09:52 +01:00
/** @var TransactionGroupTransformer $transformer */
$transformer = app(TransactionGroupTransformer::class);
2018-12-15 22:03:05 +01:00
$transformer->setParameters($this->parameters);
2019-03-30 07:09:52 +01:00
$resource = new Item($selectedGroup, $transformer, 'transactions');
2018-02-13 21:04:15 +01:00
2018-02-18 10:31:15 +01:00
return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json');
2018-02-13 21:04:15 +01:00
}
2018-03-05 19:35:58 +01:00
}