. */ declare(strict_types=1); namespace FireflyIII\Api\V1\Controllers; use FireflyIII\Api\V1\Requests\TransactionRequest; use FireflyIII\Helpers\Collector\JournalCollectorInterface; use FireflyIII\Helpers\Filter\InternalTransferFilter; use FireflyIII\Helpers\Filter\NegativeAmountFilter; use FireflyIII\Helpers\Filter\PositiveAmountFilter; use FireflyIII\Models\Transaction; use FireflyIII\Models\TransactionJournal; use FireflyIII\Models\TransactionType; use FireflyIII\Repositories\Journal\JournalRepositoryInterface; use FireflyIII\Services\Internal\Update\JournalUpdateService; use FireflyIII\Transformers\TransactionTransformer; use Illuminate\Http\Request; 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 Preferences; /** * Class TransactionController */ class TransactionController extends Controller { /** @var JournalRepositoryInterface */ private $repository; /** * TransactionController constructor. * * @throws \FireflyIII\Exceptions\FireflyException */ public function __construct() { parent::__construct(); $this->middleware( function ($request, $next) { /** @var JournalRepositoryInterface repository */ $this->repository = app(JournalRepositoryInterface::class); $this->repository->setUser(auth()->user()); return $next($request); } ); } /** * Remove the specified resource from storage. * * @param \FireflyIII\Models\Transaction $transaction * * @return \Illuminate\Http\Response */ public function delete(Transaction $transaction) { $journal = $transaction->transactionJournal; $this->repository->delete($journal); return response()->json([], 204); } /** * @param Request $request * * @return \Illuminate\Http\JsonResponse */ public function index(Request $request) { $pageSize = intval(Preferences::getForUser(auth()->user(), 'listPageSize', 50)->data); // read type from URI $type = $request->get('type') ?? 'default'; $this->parameters->set('type', $type); // types to get, page size: $types = $this->mapTypes($this->parameters->get('type')); $manager = new Manager(); $baseUrl = $request->getSchemeAndHttpHost() . '/api/v1'; $manager->setSerializer(new JsonApiSerializer($baseUrl)); // collect transactions using the journal collector $collector = app(JournalCollectorInterface::class); $collector->setUser(auth()->user()); $collector->withOpposingAccount()->withCategoryInformation()->withBudgetInformation(); $collector->setAllAssetAccounts(); // remove internal transfer filter: if (in_array(TransactionType::TRANSFER, $types)) { $collector->removeFilter(InternalTransferFilter::class); } if (!is_null($this->parameters->get('start')) && !is_null($this->parameters->get('end'))) { $collector->setRange($this->parameters->get('start'), $this->parameters->get('end')); } $collector->setLimit($pageSize)->setPage($this->parameters->get('page')); $collector->setTypes($types); $paginator = $collector->getPaginatedJournals(); $paginator->setPath(route('api.v1.transactions.index') . $this->buildParams()); $transactions = $paginator->getCollection(); $resource = new FractalCollection($transactions, new TransactionTransformer($this->parameters), 'transactions'); $resource->setPaginator(new IlluminatePaginatorAdapter($paginator)); return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json'); } /** * @param Request $request * @param Transaction $transaction * * @return \Illuminate\Http\JsonResponse */ public function show(Request $request, Transaction $transaction) { $manager = new Manager(); $baseUrl = $request->getSchemeAndHttpHost() . '/api/v1'; $manager->setSerializer(new JsonApiSerializer($baseUrl)); // add include parameter: $include = $request->get('include') ?? ''; $manager->parseIncludes($include); // collect transactions using the journal collector $collector = app(JournalCollectorInterface::class); $collector->setUser(auth()->user()); $collector->withOpposingAccount()->withCategoryInformation()->withBudgetInformation(); // filter on specific journals. $collector->setJournals(new Collection([$transaction->transactionJournal])); // add filter to remove transactions: $transactionType = $transaction->transactionJournal->transactionType->type; if ($transactionType === TransactionType::WITHDRAWAL) { $collector->addFilter(PositiveAmountFilter::class); } if (!($transactionType === TransactionType::WITHDRAWAL)) { $collector->addFilter(NegativeAmountFilter::class); } $transactions = $collector->getJournals(); $resource = new Item($transactions->first(), new TransactionTransformer($this->parameters), 'transactions'); return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json'); } /** * @param TransactionRequest $request * * @return \Illuminate\Http\JsonResponse * @throws \FireflyIII\Exceptions\FireflyException */ public function store(TransactionRequest $request, JournalRepositoryInterface $repository) { $data = $request->getAll(); $data['user'] = auth()->user()->id; $journal = $repository->store($data); $manager = new Manager(); $baseUrl = $request->getSchemeAndHttpHost() . '/api/v1'; $manager->setSerializer(new JsonApiSerializer($baseUrl)); // add include parameter: $include = $request->get('include') ?? ''; $manager->parseIncludes($include); // collect transactions using the journal collector $collector = app(JournalCollectorInterface::class); $collector->setUser(auth()->user()); $collector->withOpposingAccount()->withCategoryInformation()->withBudgetInformation(); // filter on specific journals. $collector->setJournals(new Collection([$journal])); // add filter to remove transactions: $transactionType = $journal->transactionType->type; if ($transactionType === TransactionType::WITHDRAWAL) { $collector->addFilter(PositiveAmountFilter::class); } if (!($transactionType === TransactionType::WITHDRAWAL)) { $collector->addFilter(NegativeAmountFilter::class); } $transactions = $collector->getJournals(); $resource = new FractalCollection($transactions, new TransactionTransformer($this->parameters), 'transactions'); return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json'); } /** * @param TransactionRequest $request * @param TransactionJournal $journal * * @return \Illuminate\Http\JsonResponse */ public function update(TransactionRequest $request, Transaction $transaction) { $data = $request->getAll(); $data['user'] = auth()->user()->id; /** @var JournalUpdateService $service */ $service = app(JournalUpdateService::class); $journal = $service->update($transaction->transactionJournal, $data); $manager = new Manager(); $baseUrl = $request->getSchemeAndHttpHost() . '/api/v1'; $manager->setSerializer(new JsonApiSerializer($baseUrl)); // add include parameter: $include = $request->get('include') ?? ''; $manager->parseIncludes($include); // needs a lot of extra data to match the journal collector. Or just expand that one. // collect transactions using the journal collector $collector = app(JournalCollectorInterface::class); $collector->setUser(auth()->user()); $collector->withOpposingAccount()->withCategoryInformation()->withBudgetInformation(); // filter on specific journals. $collector->setJournals(new Collection([$journal])); // add filter to remove transactions: $transactionType = $journal->transactionType->type; if ($transactionType === TransactionType::WITHDRAWAL) { $collector->addFilter(PositiveAmountFilter::class); } if (!($transactionType === TransactionType::WITHDRAWAL)) { $collector->addFilter(NegativeAmountFilter::class); } $transactions = $collector->getJournals(); $resource = new FractalCollection($transactions, new TransactionTransformer($this->parameters), 'transactions'); return response()->json($manager->createData($resource)->toArray())->header('Content-Type', 'application/vnd.api+json'); } /** * @param string $type * * @return array */ private function mapTypes(string $type): array { $types = [ 'all' => [ TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::TRANSFER, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION, ], 'withdrawal' => [ TransactionType::WITHDRAWAL, ], 'withdrawals' => [ TransactionType::WITHDRAWAL, ], 'expense' => [ TransactionType::WITHDRAWAL, ], 'income' => [ TransactionType::DEPOSIT, ], 'deposit' => [ TransactionType::DEPOSIT, ], 'deposits' => [ TransactionType::DEPOSIT, ], 'transfer' => [ TransactionType::TRANSFER, ], 'transfers' => [ TransactionType::TRANSFER, ], 'opening_balance' => [ TransactionType::OPENING_BALANCE, ], 'reconciliation' => [ TransactionType::RECONCILIATION, ], 'reconciliations' => [ TransactionType::RECONCILIATION, ], 'special' => [ TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION, ], 'specials' => [ TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION, ], 'default' => [ TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::TRANSFER, ], ]; if (isset($types[$type])) { return $types[$type]; } return $types['default']; // @codeCoverageIgnore } }