mirror of
https://github.com/firefly-iii/firefly-iii.git
synced 2025-02-25 18:45:27 -06:00
Improve test coverage.
This commit is contained in:
parent
6fdfa722dd
commit
aacd218056
@ -55,13 +55,17 @@ class MigrateAttachments extends Command
|
||||
*/
|
||||
public function handle(): int
|
||||
{
|
||||
// @codeCoverageIgnoreStart
|
||||
$start = microtime(true);
|
||||
if ($this->isExecuted() && true !== $this->option('force')) {
|
||||
$this->warn('This command has already been executed.');
|
||||
|
||||
return 0;
|
||||
}
|
||||
// @codeCoverageIgnoreEnd
|
||||
|
||||
$attachments = Attachment::get();
|
||||
$count = 0;
|
||||
|
||||
/** @var Attachment $att */
|
||||
foreach ($attachments as $att) {
|
||||
@ -69,6 +73,7 @@ class MigrateAttachments extends Command
|
||||
// move description:
|
||||
$description = (string)$att->description;
|
||||
if ('' !== $description) {
|
||||
|
||||
// find or create note:
|
||||
$note = $att->notes()->first();
|
||||
if (null === $note) {
|
||||
@ -82,9 +87,16 @@ class MigrateAttachments extends Command
|
||||
$att->description = '';
|
||||
$att->save();
|
||||
|
||||
Log::debug(sprintf('Migrated attachment #%s description to note #%d', $att->id, $note->id));
|
||||
Log::debug(sprintf('Migrated attachment #%s description to note #%d.', $att->id, $note->id));
|
||||
$count++;
|
||||
}
|
||||
}
|
||||
if (0 === $count) {
|
||||
$this->line('All attachments are OK.');
|
||||
}
|
||||
if (0 !== $count) {
|
||||
$this->line(sprintf('Updated %d attachment(s).',$count));
|
||||
}
|
||||
$end = round(microtime(true) - $start, 2);
|
||||
$this->info(sprintf('Migrated attachment notes in %s seconds.', $end));
|
||||
$this->markAsExecuted();
|
||||
|
@ -1,6 +1,6 @@
|
||||
<?php
|
||||
/**
|
||||
* MigrateNotes.php
|
||||
* MigrateJournalNotes.php
|
||||
* Copyright (c) 2019 thegrumpydictator@gmail.com
|
||||
*
|
||||
* This file is part of Firefly III.
|
||||
@ -31,9 +31,9 @@ use Illuminate\Console\Command;
|
||||
use Log;
|
||||
|
||||
/**
|
||||
* Class MigrateNotes
|
||||
* Class MigrateJournalNotes
|
||||
*/
|
||||
class MigrateNotes extends Command
|
||||
class MigrateJournalNotes extends Command
|
||||
{
|
||||
public const CONFIG_NAME = '4780_migrate_notes';
|
||||
/**
|
||||
@ -41,7 +41,7 @@ class MigrateNotes extends Command
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $description = 'Migrate notes';
|
||||
protected $description = 'Migrate notes for transaction journals.';
|
||||
/**
|
||||
* The name and signature of the console command.
|
||||
*
|
||||
@ -57,12 +57,14 @@ class MigrateNotes extends Command
|
||||
public function handle(): int
|
||||
{
|
||||
$start = microtime(true);
|
||||
// @codeCoverageIgnoreStart
|
||||
if ($this->isExecuted() && true !== $this->option('force')) {
|
||||
$this->warn('This command has already been executed.');
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
// @codeCoverageIgnoreEnd
|
||||
$count = 0;
|
||||
/** @noinspection PhpUndefinedMethodInspection */
|
||||
$set = TransactionJournalMeta::whereName('notes')->get();
|
||||
/** @var TransactionJournalMeta $meta */
|
||||
@ -79,10 +81,21 @@ class MigrateNotes extends Command
|
||||
Log::debug(sprintf('Migrated meta note #%d to Note #%d', $meta->id, $note->id));
|
||||
try {
|
||||
$meta->delete();
|
||||
// @codeCoverageIgnoreStart
|
||||
} catch (Exception $e) {
|
||||
Log::error(sprintf('Could not delete old meta entry #%d: %s', $meta->id, $e->getMessage()));
|
||||
}
|
||||
// @codeCoverageIgnoreEnd
|
||||
$count++;
|
||||
}
|
||||
|
||||
if (0 === $count) {
|
||||
$this->line('No notes to migrate.');
|
||||
}
|
||||
if (0 !== $count) {
|
||||
$this->line(sprintf('Migrated %d note(s).', $count));
|
||||
}
|
||||
|
||||
$end = round(microtime(true) - $start, 2);
|
||||
$this->info(sprintf('Migrated notes in %s seconds.', $end));
|
||||
$this->markAsExecuted();
|
@ -60,6 +60,7 @@ class MigrateToGroups extends Command
|
||||
private $journalRepository;
|
||||
/** @var JournalDestroyService */
|
||||
private $service;
|
||||
private $count;
|
||||
|
||||
/**
|
||||
* Create a new command instance.
|
||||
@ -69,6 +70,7 @@ class MigrateToGroups extends Command
|
||||
public function __construct()
|
||||
{
|
||||
parent::__construct();
|
||||
$this->count = 0;
|
||||
$this->journalRepository = app(JournalRepositoryInterface::class);
|
||||
$this->service = app(JournalDestroyService::class);
|
||||
$this->groupFactory = app(TransactionGroupFactory::class);
|
||||
@ -83,14 +85,17 @@ class MigrateToGroups extends Command
|
||||
public function handle(): int
|
||||
{
|
||||
$start = microtime(true);
|
||||
// @codeCoverageIgnoreStart
|
||||
if ($this->isMigrated() && true !== $this->option('force')) {
|
||||
$this->info('Database already seems to be migrated.');
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
if (true === $this->option('force')) {
|
||||
$this->warn('Forcing the migration.');
|
||||
}
|
||||
// @codeCoverageIgnoreEnd
|
||||
|
||||
Log::debug('---- start group migration ----');
|
||||
$this->makeGroupsFromSplitJournals();
|
||||
@ -102,14 +107,24 @@ class MigrateToGroups extends Command
|
||||
Log::debug('---- end group migration ----');
|
||||
$end = round(microtime(true) - $start, 2);
|
||||
$this->info(sprintf('Migrate all journals to groups in %s seconds.', $end));
|
||||
|
||||
if (0 !== $this->count) {
|
||||
$this->line(sprintf('Migrated %d transaction journal(s).', $this->count));
|
||||
}
|
||||
if (0 === $this->count) {
|
||||
$this->line('No journals to migrate to groups.');
|
||||
}
|
||||
|
||||
|
||||
$this->markAsMigrated();
|
||||
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param TransactionJournal $journal
|
||||
* @param Transaction $transaction
|
||||
* @param Transaction $transaction
|
||||
*
|
||||
* @return Transaction|null
|
||||
*/
|
||||
@ -117,7 +132,12 @@ class MigrateToGroups extends Command
|
||||
{
|
||||
$set = $journal->transactions->filter(
|
||||
static function (Transaction $subject) use ($transaction) {
|
||||
return $transaction->amount * -1 === (float)$subject->amount && $transaction->identifier === $subject->identifier;
|
||||
$amount = (float)$transaction->amount * -1 === (float)$subject->amount;
|
||||
$identifier = $transaction->identifier === $subject->identifier;
|
||||
Log::debug(sprintf('Amount the same? %s', var_export($amount, true)));
|
||||
Log::debug(sprintf('ID the same? %s', var_export($identifier, true)));
|
||||
|
||||
return $amount && $identifier;
|
||||
}
|
||||
);
|
||||
|
||||
@ -152,6 +172,7 @@ class MigrateToGroups extends Command
|
||||
]
|
||||
);
|
||||
DB::table('transaction_journals')->where('id', $array['id'])->update(['transaction_group_id' => $groupId]);
|
||||
$this->count++;
|
||||
}
|
||||
|
||||
/**
|
||||
@ -209,14 +230,17 @@ class MigrateToGroups extends Command
|
||||
* @param TransactionJournal $journal
|
||||
*
|
||||
* @throws Exception
|
||||
* @SuppressWarnings(PHPMD.ExcessiveMethodLength)
|
||||
*/
|
||||
private function makeMultiGroup(TransactionJournal $journal): void
|
||||
{
|
||||
// double check transaction count.
|
||||
if ($journal->transactions->count() <= 2) {
|
||||
// @codeCoverageIgnoreStart
|
||||
Log::debug(sprintf('Will not try to convert journal #%d because it has 2 or less transactions.', $journal->id));
|
||||
|
||||
return;
|
||||
// @codeCoverageIgnoreEnd
|
||||
}
|
||||
Log::debug(sprintf('Will now try to convert journal #%d', $journal->id));
|
||||
|
||||
@ -264,6 +288,7 @@ class MigrateToGroups extends Command
|
||||
$opposingTr = $this->findOpposingTransaction($journal, $transaction);
|
||||
|
||||
if (null === $opposingTr) {
|
||||
// @codeCoverageIgnoreStart
|
||||
$this->error(
|
||||
sprintf(
|
||||
'Journal #%d has no opposing transaction for transaction #%d. Cannot upgrade this entry.',
|
||||
@ -271,6 +296,7 @@ class MigrateToGroups extends Command
|
||||
)
|
||||
);
|
||||
continue;
|
||||
// @codeCoverageIgnoreEnd
|
||||
}
|
||||
|
||||
$tArray = [
|
||||
@ -321,12 +347,16 @@ class MigrateToGroups extends Command
|
||||
// delete the old transaction journal.
|
||||
$this->service->destroy($journal);
|
||||
|
||||
$this->count++;
|
||||
|
||||
// report on result:
|
||||
Log::debug(
|
||||
sprintf('Migrated journal #%d into group #%d with these journals: #%s', $journal->id, $group->id, implode(', #', $group->transactionJournals->pluck('id')->toArray()))
|
||||
sprintf('Migrated journal #%d into group #%d with these journals: #%s',
|
||||
$journal->id, $group->id, implode(', #', $group->transactionJournals->pluck('id')->toArray()))
|
||||
);
|
||||
$this->line(
|
||||
sprintf('Migrated journal #%d into group #%d with these journals: #%s', $journal->id, $group->id, implode(', #', $group->transactionJournals->pluck('id')->toArray()))
|
||||
sprintf('Migrated journal #%d into group #%d with these journals: #%s',
|
||||
$journal->id, $group->id, implode(', #', $group->transactionJournals->pluck('id')->toArray()))
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -24,19 +24,16 @@ declare(strict_types=1);
|
||||
namespace FireflyIII\Console\Commands\Upgrade;
|
||||
|
||||
|
||||
use Crypt;
|
||||
use FireflyIII\Exceptions\FireflyException;
|
||||
use FireflyIII\Models\Bill;
|
||||
use FireflyIII\Models\Preference;
|
||||
use FireflyIII\Models\Rule;
|
||||
use FireflyIII\Models\RuleAction;
|
||||
use FireflyIII\Models\RuleGroup;
|
||||
use FireflyIII\Models\RuleTrigger;
|
||||
use FireflyIII\Models\TransactionCurrency;
|
||||
use FireflyIII\Repositories\Bill\BillRepositoryInterface;
|
||||
use FireflyIII\Repositories\Rule\RuleRepositoryInterface;
|
||||
use FireflyIII\Repositories\RuleGroup\RuleGroupRepositoryInterface;
|
||||
use FireflyIII\Repositories\User\UserRepositoryInterface;
|
||||
use FireflyIII\User;
|
||||
use Illuminate\Console\Command;
|
||||
use Illuminate\Contracts\Encryption\DecryptException;
|
||||
use Illuminate\Support\Collection;
|
||||
use Log;
|
||||
|
||||
/**
|
||||
* Class MigrateToRules
|
||||
@ -57,171 +54,60 @@ class MigrateToRules extends Command
|
||||
*/
|
||||
protected $signature = 'firefly-iii:bills-to-rules {--F|force : Force the execution of this command.}';
|
||||
|
||||
/** @var UserRepositoryInterface */
|
||||
private $userRepository;
|
||||
/** @var RuleGroupRepositoryInterface */
|
||||
private $ruleGroupRepository;
|
||||
/** @var BillRepositoryInterface */
|
||||
private $billRepository;
|
||||
/** @var RuleRepositoryInterface */
|
||||
private $ruleRepository;
|
||||
private $count;
|
||||
|
||||
/**
|
||||
* MigrateToRules constructor.
|
||||
*/
|
||||
public function __construct()
|
||||
{
|
||||
parent::__construct();
|
||||
$this->count = 0;
|
||||
$this->userRepository = app(UserRepositoryInterface::class);
|
||||
$this->ruleGroupRepository = app(RuleGroupRepositoryInterface::class);
|
||||
$this->billRepository = app(BillRepositoryInterface::class);
|
||||
$this->ruleRepository = app(RuleRepositoryInterface::class);
|
||||
}
|
||||
|
||||
/**
|
||||
* Execute the console command.
|
||||
*
|
||||
* @return int
|
||||
* @throws FireflyException
|
||||
*/
|
||||
public function handle(): int
|
||||
{
|
||||
$start = microtime(true);
|
||||
$start = microtime(true);
|
||||
|
||||
// @codeCoverageIgnoreStart
|
||||
if ($this->isExecuted() && true !== $this->option('force')) {
|
||||
$this->warn('This command has already been executed.');
|
||||
|
||||
return 0;
|
||||
}
|
||||
// @codeCoverageIgnoreEnd
|
||||
|
||||
foreach (User::get() as $user) {
|
||||
/** @var Preference $lang */
|
||||
$lang = app('preferences')->getForUser($user, 'language', 'en_US');
|
||||
$groupName = (string)trans('firefly.rulegroup_for_bills_title', [], $lang->data);
|
||||
$ruleGroup = $user->ruleGroups()->where('title', $groupName)->first();
|
||||
$currencyPreference = app('preferences')->getForUser($user, 'currencyPreference', config('firefly.default_currency', 'EUR'));
|
||||
|
||||
if (null === $currencyPreference) {
|
||||
$this->error('User has no currency preference. Impossible.');
|
||||
|
||||
return 1;
|
||||
}
|
||||
$currencyCode = $this->tryDecrypt($currencyPreference->data);
|
||||
|
||||
// try json decrypt just in case.
|
||||
if (strlen($currencyCode) > 3) {
|
||||
$currencyCode = json_decode($currencyCode) ?? 'EUR';
|
||||
}
|
||||
|
||||
$currency = TransactionCurrency::where('code', $currencyCode)->first();
|
||||
if (null === $currency) {
|
||||
$this->line('Fall back to default currency in migrateBillsToRules().');
|
||||
$currency = app('amount')->getDefaultCurrencyByUser($user);
|
||||
}
|
||||
|
||||
if (null === $ruleGroup) {
|
||||
$array = RuleGroup::get(['order'])->pluck('order')->toArray();
|
||||
$order = count($array) > 0 ? max($array) + 1 : 1;
|
||||
$ruleGroup = RuleGroup::create(
|
||||
[
|
||||
'user_id' => $user->id,
|
||||
'title' => (string)trans('firefly.rulegroup_for_bills_title', [], $lang->data),
|
||||
'description' => (string)trans('firefly.rulegroup_for_bills_description', [], $lang->data),
|
||||
'order' => $order,
|
||||
'active' => 1,
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
// loop bills.
|
||||
$order = 1;
|
||||
$count = 0;
|
||||
/** @var Collection $collection */
|
||||
$collection = $user->bills()->get();
|
||||
/** @var Bill $bill */
|
||||
foreach ($collection as $bill) {
|
||||
if ('MIGRATED_TO_RULES' !== $bill->match) {
|
||||
$count++;
|
||||
$rule = Rule::create(
|
||||
[
|
||||
'user_id' => $user->id,
|
||||
'rule_group_id' => $ruleGroup->id,
|
||||
'title' => (string)trans('firefly.rule_for_bill_title', ['name' => $bill->name], $lang->data),
|
||||
'description' => (string)trans('firefly.rule_for_bill_description', ['name' => $bill->name], $lang->data),
|
||||
'order' => $order,
|
||||
'active' => $bill->active,
|
||||
'stop_processing' => 1,
|
||||
]
|
||||
);
|
||||
// add default trigger
|
||||
RuleTrigger::create(
|
||||
[
|
||||
'rule_id' => $rule->id,
|
||||
'trigger_type' => 'user_action',
|
||||
'trigger_value' => 'store-journal',
|
||||
'active' => 1,
|
||||
'stop_processing' => 0,
|
||||
'order' => 1,
|
||||
]
|
||||
);
|
||||
// add trigger for description
|
||||
$match = implode(' ', explode(',', $bill->match));
|
||||
RuleTrigger::create(
|
||||
[
|
||||
'rule_id' => $rule->id,
|
||||
'trigger_type' => 'description_contains',
|
||||
'trigger_value' => $match,
|
||||
'active' => 1,
|
||||
'stop_processing' => 0,
|
||||
'order' => 2,
|
||||
]
|
||||
);
|
||||
if ($bill->amount_max !== $bill->amount_min) {
|
||||
// add triggers for amounts:
|
||||
RuleTrigger::create(
|
||||
[
|
||||
'rule_id' => $rule->id,
|
||||
'trigger_type' => 'amount_less',
|
||||
'trigger_value' => round($bill->amount_max, $currency->decimal_places),
|
||||
'active' => 1,
|
||||
'stop_processing' => 0,
|
||||
'order' => 3,
|
||||
]
|
||||
);
|
||||
RuleTrigger::create(
|
||||
[
|
||||
'rule_id' => $rule->id,
|
||||
'trigger_type' => 'amount_more',
|
||||
'trigger_value' => round((float)$bill->amount_min, $currency->decimal_places),
|
||||
'active' => 1,
|
||||
'stop_processing' => 0,
|
||||
'order' => 4,
|
||||
]
|
||||
);
|
||||
}
|
||||
if ($bill->amount_max === $bill->amount_min) {
|
||||
RuleTrigger::create(
|
||||
[
|
||||
'rule_id' => $rule->id,
|
||||
'trigger_type' => 'amount_exactly',
|
||||
'trigger_value' => round((float)$bill->amount_min, $currency->decimal_places),
|
||||
'active' => 1,
|
||||
'stop_processing' => 0,
|
||||
'order' => 3,
|
||||
]
|
||||
);
|
||||
}
|
||||
|
||||
// create action
|
||||
RuleAction::create(
|
||||
[
|
||||
'rule_id' => $rule->id,
|
||||
'action_type' => 'link_to_bill',
|
||||
'action_value' => $bill->name,
|
||||
'order' => 1,
|
||||
'active' => 1,
|
||||
'stop_processing' => 0,
|
||||
]
|
||||
);
|
||||
|
||||
$order++;
|
||||
$bill->match = 'MIGRATED_TO_RULES';
|
||||
$bill->save();
|
||||
$this->line(sprintf('Updated bill #%d ("%s") so it will use rules.', $bill->id, $bill->name));
|
||||
}
|
||||
|
||||
// give bills a currency when they dont have one.
|
||||
if (null === $bill->transaction_currency_id) {
|
||||
$this->line(sprintf('Gave bill #%d ("%s") a currency (%s).', $bill->id, $bill->name, $currency->name));
|
||||
$bill->transactionCurrency()->associate($currency);
|
||||
$bill->save();
|
||||
}
|
||||
}
|
||||
if ($count > 0) {
|
||||
$this->info(sprintf('Migrated %d bills for user %s', $count, $user->email));
|
||||
}
|
||||
if (0 === $count) {
|
||||
$this->info(sprintf('Bills are correct for user %s.', $user->email));
|
||||
}
|
||||
$users = $this->userRepository->all();
|
||||
/** @var User $user */
|
||||
foreach ($users as $user) {
|
||||
$this->migrateUser($user);
|
||||
}
|
||||
|
||||
if (0 === $this->count) {
|
||||
$this->line('All bills are OK.');
|
||||
}
|
||||
if (0 !== $this->count) {
|
||||
$this->line(sprintf('Verified and fixed %d bill(s).', $this->count));
|
||||
}
|
||||
|
||||
$end = round(microtime(true) - $start, 2);
|
||||
$this->info(sprintf('Verified and fixed bills in %s seconds.', $end));
|
||||
$this->markAsExecuted();
|
||||
@ -251,18 +137,107 @@ class MigrateToRules extends Command
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $value
|
||||
* Migrate bills to new rule structure for a specific user.
|
||||
*
|
||||
* @return string
|
||||
* @param User $user
|
||||
* @throws FireflyException
|
||||
*/
|
||||
private function tryDecrypt(string $value): string
|
||||
private function migrateUser(User $user): void
|
||||
{
|
||||
try {
|
||||
$value = Crypt::decrypt($value);
|
||||
} catch (DecryptException $e) {
|
||||
Log::debug(sprintf('Could not decrypt. %s', $e->getMessage()));
|
||||
$this->ruleGroupRepository->setUser($user);
|
||||
$this->billRepository->setUser($user);
|
||||
$this->ruleRepository->setUser($user);
|
||||
|
||||
/** @var Preference $lang */
|
||||
$lang = app('preferences')->getForUser($user, 'language', 'en_US');
|
||||
$groupTitle = (string)trans('firefly.rulegroup_for_bills_title', [], $lang->data);
|
||||
$ruleGroup = $this->ruleGroupRepository->findByTitle($groupTitle);
|
||||
//$currency = $this->getCurrency($user);
|
||||
|
||||
if (null === $ruleGroup) {
|
||||
$ruleGroup = $this->ruleGroupRepository->store(
|
||||
[
|
||||
'title' => (string)trans('firefly.rulegroup_for_bills_title', [], $lang->data),
|
||||
'description' => (string)trans('firefly.rulegroup_for_bills_description', [], $lang->data),
|
||||
'active' => true,
|
||||
]
|
||||
);
|
||||
}
|
||||
$bills = $this->billRepository->getBills();
|
||||
|
||||
/** @var Bill $bill */
|
||||
foreach ($bills as $bill) {
|
||||
$this->migrateBill($ruleGroup, $bill, $lang);
|
||||
}
|
||||
|
||||
return $value;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param RuleGroup $ruleGroup
|
||||
* @param Bill $bill
|
||||
* @throws FireflyException
|
||||
*/
|
||||
private function migrateBill(RuleGroup $ruleGroup, Bill $bill, Preference $language): void
|
||||
{
|
||||
if ('MIGRATED_TO_RULES' === $bill->match) {
|
||||
return;
|
||||
}
|
||||
|
||||
// get match thing:
|
||||
$match = implode(' ', explode(',', $bill->match));
|
||||
$newRule = [
|
||||
'rule_group_id' => $ruleGroup->id,
|
||||
'active' => true,
|
||||
'strict' => false,
|
||||
'stop_processing' => false, // field is no longer used.
|
||||
'title' => (string)trans('firefly.rule_for_bill_title', ['name' => $bill->name], $language->data),
|
||||
'description' => (string)trans('firefly.rule_for_bill_description', ['name' => $bill->name], $language->data),
|
||||
'trigger' => 'store-journal',
|
||||
'triggers' => [
|
||||
[
|
||||
'type' => 'description_contains',
|
||||
'value' => $match,
|
||||
],
|
||||
],
|
||||
'actions' => [
|
||||
'type' => 'link_to_bill',
|
||||
'value' => $bill->name,
|
||||
],
|
||||
];
|
||||
|
||||
// two triggers or one, depends on bill content:
|
||||
if ($bill->amount_max === $bill->amount_min) {
|
||||
$newRule['triggers'][] = [
|
||||
'type' => 'amount_exactly',
|
||||
'value' => $bill->amount_min,
|
||||
];
|
||||
}
|
||||
if ($bill->amount_max !== $bill->amount_min) {
|
||||
$newRule['triggers'][] = [
|
||||
'type' => 'amount_less',
|
||||
'value' => $bill->amount_max,
|
||||
];
|
||||
$newRule['triggers'][] = [
|
||||
'type' => 'amount_more',
|
||||
'value' => $bill->amount_min,
|
||||
];
|
||||
}
|
||||
|
||||
$this->ruleRepository->store($newRule);
|
||||
|
||||
// update bill:
|
||||
$newBillData = [
|
||||
'currency_id' => $bill->transaction_currency_id,
|
||||
'name' => $bill->name,
|
||||
'match' => 'MIGRATED_TO_RULES',
|
||||
'amount_min' => $bill->amount_min,
|
||||
'amount_max' => $bill->amount_max,
|
||||
'date' => $bill->date,
|
||||
'repeat_freq' => $bill->repeat_freq,
|
||||
'skip' => $bill->skip,
|
||||
'active' => $bill->active,
|
||||
];
|
||||
$this->billRepository->update($bill, $newBillData);
|
||||
$this->count++;
|
||||
}
|
||||
}
|
@ -84,7 +84,7 @@ class TransactionJournalMeta extends Model
|
||||
*/
|
||||
public function getDataAttribute($value)
|
||||
{
|
||||
return json_decode($value);
|
||||
return json_decode($value, false);
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -343,4 +343,14 @@ class RuleGroupRepository implements RuleGroupRepositoryInterface
|
||||
|
||||
return $ruleGroup;
|
||||
}
|
||||
|
||||
/**
|
||||
* @param string $title
|
||||
*
|
||||
* @return RuleGroup|null
|
||||
*/
|
||||
public function findByTitle(string $title): ?RuleGroup
|
||||
{
|
||||
return $this->user->ruleGroups()->where('title', $title)->first();
|
||||
}
|
||||
}
|
||||
|
@ -37,7 +37,7 @@ interface RuleGroupRepositoryInterface
|
||||
public function count(): int;
|
||||
|
||||
/**
|
||||
* @param RuleGroup $ruleGroup
|
||||
* @param RuleGroup $ruleGroup
|
||||
* @param RuleGroup|null $moveTo
|
||||
*
|
||||
* @return bool
|
||||
@ -51,6 +51,13 @@ interface RuleGroupRepositoryInterface
|
||||
*/
|
||||
public function find(int $ruleGroupId): ?RuleGroup;
|
||||
|
||||
/**
|
||||
* @param string $title
|
||||
*
|
||||
* @return RuleGroup|null
|
||||
*/
|
||||
public function findByTitle(string $title): ?RuleGroup;
|
||||
|
||||
/**
|
||||
* Get all rule groups.
|
||||
*
|
||||
@ -143,7 +150,7 @@ interface RuleGroupRepositoryInterface
|
||||
|
||||
/**
|
||||
* @param RuleGroup $ruleGroup
|
||||
* @param array $data
|
||||
* @param array $data
|
||||
*
|
||||
* @return RuleGroup
|
||||
*/
|
||||
|
@ -48,7 +48,7 @@ class BillUpdateService
|
||||
}
|
||||
|
||||
/**
|
||||
* @param Bill $bill
|
||||
* @param Bill $bill
|
||||
* @param array $data
|
||||
*
|
||||
* @return Bill
|
||||
@ -69,9 +69,9 @@ class BillUpdateService
|
||||
$currency->enabled = true;
|
||||
$currency->save();
|
||||
|
||||
|
||||
$oldName = $bill->name;
|
||||
$bill->name = $data['name'];
|
||||
$bill->match = $data['match'] ?? $bill->match;
|
||||
$bill->amount_min = $data['amount_min'];
|
||||
$bill->amount_max = $data['amount_max'];
|
||||
$bill->date = $data['date'];
|
||||
|
@ -49,306 +49,306 @@ class JournalCurrenciesTest extends TestCase
|
||||
parent::setUp();
|
||||
Log::info(sprintf('Now in %s.', get_class($this)));
|
||||
}
|
||||
|
||||
/**
|
||||
* Basic run. Would not change anything.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
*/
|
||||
public function testHandle(): void
|
||||
{
|
||||
// mock classes
|
||||
$accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
$currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$euro = TransactionCurrency::find(1);
|
||||
|
||||
// update transfer if necessary for the test:
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
|
||||
// mock stuff
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::TRANSFER]])
|
||||
->andReturn(new Collection);
|
||||
|
||||
// for the "other journals" check, return nothing.
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
->andReturn(new Collection);
|
||||
|
||||
// transaction would be verified, nothing more.
|
||||
$this->artisan('firefly-iii:journal-currencies')
|
||||
->expectsOutput('All transactions are correct.')
|
||||
->assertExitCode(0);
|
||||
// nothing changed, so no verification.
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Submit a single transfer which has no issues.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
*/
|
||||
public function testHandleTransfer(): void
|
||||
{
|
||||
// mock classes
|
||||
$accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
$currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$euro = TransactionCurrency::find(1);
|
||||
$transfer = $this->getRandomTransfer();
|
||||
|
||||
// update transfer if necessary for the test:
|
||||
$collection = new Collection([$transfer]);
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
|
||||
// mock stuff
|
||||
$accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
// return single tranfer
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::TRANSFER]])
|
||||
->andReturn($collection);
|
||||
|
||||
// for the "other journals" check, return nothing.
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
->andReturn(new Collection);
|
||||
|
||||
$accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturn($euro->id);
|
||||
$currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturn($euro);
|
||||
|
||||
// transaction would be verified, nothing more.
|
||||
$this->artisan('firefly-iii:journal-currencies')
|
||||
->expectsOutput('Verified 1 transaction(s) and journal(s).')
|
||||
->assertExitCode(0);
|
||||
// nothing changed, so no verification.
|
||||
}
|
||||
|
||||
/**
|
||||
* Submit a single transfer where the source account has no currency preference.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
*/
|
||||
public function testHandleTransferSourceNoPref(): void
|
||||
{
|
||||
// mock classes
|
||||
$accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
$currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$euro = TransactionCurrency::find(1);
|
||||
$transfer = $this->getRandomTransfer();
|
||||
|
||||
// edit source to remove currency preference:
|
||||
/** @var Account $source */
|
||||
$source = $transfer->transactions()->where('amount', '<', 0)->first()->account;
|
||||
// AccountMeta::where('account_id', $source->id)->where('name', 'currency_id')->delete();
|
||||
|
||||
// update transfer if necessary for the test:
|
||||
$collection = new Collection([$transfer]);
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
|
||||
// mock stuff
|
||||
$accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
// return single transfer
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::TRANSFER]])
|
||||
->andReturn($collection);
|
||||
|
||||
// for the "other journals" check, return nothing.
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
->andReturn(new Collection);
|
||||
|
||||
// return NULL for first currency ID and currency.
|
||||
$accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturnNull();
|
||||
$currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturnNull();
|
||||
|
||||
// transaction would be verified, nothing more.
|
||||
$this->artisan('firefly-iii:journal-currencies')
|
||||
->expectsOutput(sprintf('Account #%d ("%s") must have currency preference but has none.', $source->id, $source->name))
|
||||
->assertExitCode(0);
|
||||
// nothing changed, so no verification.
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Submit a single transfer where the source transaction has no currency set.
|
||||
* Because this is not done over repositories, we must edit the DB.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
*/
|
||||
public function testHandleTransferSourceNoCurrency(): void
|
||||
{
|
||||
// mock classes
|
||||
$accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
$currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$euro = TransactionCurrency::find(1);
|
||||
$transfer = $this->getRandomTransfer();
|
||||
/** @var Transaction $source */
|
||||
$source = $transfer->transactions()->where('amount', '<', 0)->first();
|
||||
$source->transaction_currency_id = null;
|
||||
$source->save();
|
||||
|
||||
// update transfer if necessary for the test:
|
||||
$collection = new Collection([$transfer]);
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
|
||||
// mock stuff
|
||||
$accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
// return single tranfer
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::TRANSFER]])
|
||||
->andReturn($collection);
|
||||
|
||||
// for the "other journals" check, return nothing.
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
->andReturn(new Collection);
|
||||
|
||||
$accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturn($euro->id);
|
||||
$currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturn($euro);
|
||||
|
||||
// transaction would be verified, nothing more.
|
||||
$this->artisan('firefly-iii:journal-currencies')
|
||||
->expectsOutput(sprintf('Transaction #%d has no currency setting, now set to %s.', $source->id, $euro->code))
|
||||
->expectsOutput('Verified 2 transaction(s) and journal(s).')
|
||||
->assertExitCode(0);
|
||||
|
||||
// check transaction
|
||||
$this->assertCount(1, Transaction::where('id', $source->id)->where('transaction_currency_id', $euro->id)->get());
|
||||
}
|
||||
|
||||
/**
|
||||
* Submit a single transfer where the source transaction has a different currency than the source account does.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
*/
|
||||
public function testHandleMismatchedTransfer(): void
|
||||
{
|
||||
// mock classes
|
||||
$accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
$currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$euro = TransactionCurrency::find(1);
|
||||
$usd = TransactionCurrency::where('code', 'USD')->first();
|
||||
$transfer = $this->getRandomTransfer();
|
||||
|
||||
/** @var Transaction $source */
|
||||
$source = $transfer->transactions()->where('amount', '<', 0)->first();
|
||||
$source->transaction_currency_id = $usd->id;
|
||||
$source->save();
|
||||
|
||||
// update transfer if necessary for the test:
|
||||
$collection = new Collection([$transfer]);
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
|
||||
// mock stuff
|
||||
$accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
// return single tranfer
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::TRANSFER]])
|
||||
->andReturn($collection);
|
||||
|
||||
// for the "other journals" check, return nothing.
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
->andReturn(new Collection);
|
||||
|
||||
$accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturn($euro->id);
|
||||
$currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturn($euro);
|
||||
|
||||
// transaction would be verified, nothing more.
|
||||
$this->artisan('firefly-iii:journal-currencies')
|
||||
->expectsOutput(
|
||||
sprintf(
|
||||
'Transaction #%d has a currency setting #%d that should be #%d. Amount remains %s, currency is changed.',
|
||||
$source->id,
|
||||
$source->transaction_currency_id,
|
||||
$euro->id,
|
||||
$source->amount
|
||||
)
|
||||
)
|
||||
->expectsOutput('Verified 2 transaction(s) and journal(s).')
|
||||
->assertExitCode(0);
|
||||
// nothing changed, so no verification.
|
||||
}
|
||||
|
||||
/**
|
||||
* Submit a single transfer where the destination account has no currency preference.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
*/
|
||||
public function testHandleTransferNoDestinationCurrency(): void
|
||||
{
|
||||
// mock classes
|
||||
$accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
$currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$euro = TransactionCurrency::find(1);
|
||||
$transfer = $this->getRandomTransfer();
|
||||
|
||||
/** @var Account $destination */
|
||||
$destination = $transfer->transactions()->where('amount', '>', 0)->first()->account;
|
||||
|
||||
// update transfer if necessary for the test:
|
||||
$collection = new Collection([$transfer]);
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
|
||||
// mock stuff
|
||||
$accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
// return single tranfer
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::TRANSFER]])
|
||||
->andReturn($collection);
|
||||
|
||||
// for the "other journals" check, return nothing.
|
||||
$journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
->andReturn(new Collection);
|
||||
|
||||
$accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturn($euro->id, 0);
|
||||
$currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturn($euro, null);
|
||||
|
||||
// transaction would be verified, nothing more.
|
||||
$this->artisan('firefly-iii:journal-currencies')
|
||||
->expectsOutput(sprintf('Account #%d ("%s") must have currency preference but has none.', $destination->id, $destination->name))
|
||||
->assertExitCode(0);
|
||||
// nothing changed, so no verification.
|
||||
}
|
||||
//
|
||||
// /**
|
||||
// * Basic run. Would not change anything.
|
||||
// *
|
||||
// * @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
// */
|
||||
// public function testHandle(): void
|
||||
// {
|
||||
// // mock classes
|
||||
// $accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
// $currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
// $journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
// $euro = TransactionCurrency::find(1);
|
||||
//
|
||||
// // update transfer if necessary for the test:
|
||||
// $false = new Configuration;
|
||||
// $false->data = false;
|
||||
// FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
// FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
//
|
||||
// // mock stuff
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::TRANSFER]])
|
||||
// ->andReturn(new Collection);
|
||||
//
|
||||
// // for the "other journals" check, return nothing.
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
// ->andReturn(new Collection);
|
||||
//
|
||||
// // transaction would be verified, nothing more.
|
||||
// $this->artisan('firefly-iii:journal-currencies')
|
||||
// ->expectsOutput('All transactions are correct.')
|
||||
// ->assertExitCode(0);
|
||||
// // nothing changed, so no verification.
|
||||
//
|
||||
// }
|
||||
//
|
||||
// /**
|
||||
// * Submit a single transfer which has no issues.
|
||||
// *
|
||||
// * @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
// */
|
||||
// public function testHandleTransfer(): void
|
||||
// {
|
||||
// // mock classes
|
||||
// $accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
// $currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
// $journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
// $euro = TransactionCurrency::find(1);
|
||||
// $transfer = $this->getRandomTransfer();
|
||||
//
|
||||
// // update transfer if necessary for the test:
|
||||
// $collection = new Collection([$transfer]);
|
||||
// $false = new Configuration;
|
||||
// $false->data = false;
|
||||
// FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
// FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
//
|
||||
// // mock stuff
|
||||
// $accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
//
|
||||
// // return single tranfer
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::TRANSFER]])
|
||||
// ->andReturn($collection);
|
||||
//
|
||||
// // for the "other journals" check, return nothing.
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
// ->andReturn(new Collection);
|
||||
//
|
||||
// $accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturn($euro->id);
|
||||
// $currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturn($euro);
|
||||
//
|
||||
// // transaction would be verified, nothing more.
|
||||
// $this->artisan('firefly-iii:journal-currencies')
|
||||
// ->expectsOutput('Verified 1 transaction(s) and journal(s).')
|
||||
// ->assertExitCode(0);
|
||||
// // nothing changed, so no verification.
|
||||
// }
|
||||
//
|
||||
// /**
|
||||
// * Submit a single transfer where the source account has no currency preference.
|
||||
// *
|
||||
// * @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
// */
|
||||
// public function testHandleTransferSourceNoPref(): void
|
||||
// {
|
||||
// // mock classes
|
||||
// $accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
// $currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
// $journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
// $euro = TransactionCurrency::find(1);
|
||||
// $transfer = $this->getRandomTransfer();
|
||||
//
|
||||
// // edit source to remove currency preference:
|
||||
// /** @var Account $source */
|
||||
// $source = $transfer->transactions()->where('amount', '<', 0)->first()->account;
|
||||
//// AccountMeta::where('account_id', $source->id)->where('name', 'currency_id')->delete();
|
||||
//
|
||||
// // update transfer if necessary for the test:
|
||||
// $collection = new Collection([$transfer]);
|
||||
// $false = new Configuration;
|
||||
// $false->data = false;
|
||||
// FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
// FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
//
|
||||
// // mock stuff
|
||||
// $accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
//
|
||||
// // return single transfer
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::TRANSFER]])
|
||||
// ->andReturn($collection);
|
||||
//
|
||||
// // for the "other journals" check, return nothing.
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
// ->andReturn(new Collection);
|
||||
//
|
||||
// // return NULL for first currency ID and currency.
|
||||
// $accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturnNull();
|
||||
// $currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturnNull();
|
||||
//
|
||||
// // transaction would be verified, nothing more.
|
||||
// $this->artisan('firefly-iii:journal-currencies')
|
||||
// ->expectsOutput(sprintf('Account #%d ("%s") must have currency preference but has none.', $source->id, $source->name))
|
||||
// ->assertExitCode(0);
|
||||
// // nothing changed, so no verification.
|
||||
//
|
||||
// }
|
||||
//
|
||||
// /**
|
||||
// * Submit a single transfer where the source transaction has no currency set.
|
||||
// * Because this is not done over repositories, we must edit the DB.
|
||||
// *
|
||||
// * @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
// */
|
||||
// public function testHandleTransferSourceNoCurrency(): void
|
||||
// {
|
||||
// // mock classes
|
||||
// $accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
// $currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
// $journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
// $euro = TransactionCurrency::find(1);
|
||||
// $transfer = $this->getRandomTransfer();
|
||||
// /** @var Transaction $source */
|
||||
// $source = $transfer->transactions()->where('amount', '<', 0)->first();
|
||||
// $source->transaction_currency_id = null;
|
||||
// $source->save();
|
||||
//
|
||||
// // update transfer if necessary for the test:
|
||||
// $collection = new Collection([$transfer]);
|
||||
// $false = new Configuration;
|
||||
// $false->data = false;
|
||||
// FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
// FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
//
|
||||
// // mock stuff
|
||||
// $accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
//
|
||||
// // return single tranfer
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::TRANSFER]])
|
||||
// ->andReturn($collection);
|
||||
//
|
||||
// // for the "other journals" check, return nothing.
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
// ->andReturn(new Collection);
|
||||
//
|
||||
// $accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturn($euro->id);
|
||||
// $currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturn($euro);
|
||||
//
|
||||
// // transaction would be verified, nothing more.
|
||||
// $this->artisan('firefly-iii:journal-currencies')
|
||||
// ->expectsOutput(sprintf('Transaction #%d has no currency setting, now set to %s.', $source->id, $euro->code))
|
||||
// ->expectsOutput('Verified 2 transaction(s) and journal(s).')
|
||||
// ->assertExitCode(0);
|
||||
//
|
||||
// // check transaction
|
||||
// $this->assertCount(1, Transaction::where('id', $source->id)->where('transaction_currency_id', $euro->id)->get());
|
||||
// }
|
||||
//
|
||||
// /**
|
||||
// * Submit a single transfer where the source transaction has a different currency than the source account does.
|
||||
// *
|
||||
// * @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
// */
|
||||
// public function testHandleMismatchedTransfer(): void
|
||||
// {
|
||||
// // mock classes
|
||||
// $accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
// $currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
// $journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
// $euro = TransactionCurrency::find(1);
|
||||
// $usd = TransactionCurrency::where('code', 'USD')->first();
|
||||
// $transfer = $this->getRandomTransfer();
|
||||
//
|
||||
// /** @var Transaction $source */
|
||||
// $source = $transfer->transactions()->where('amount', '<', 0)->first();
|
||||
// $source->transaction_currency_id = $usd->id;
|
||||
// $source->save();
|
||||
//
|
||||
// // update transfer if necessary for the test:
|
||||
// $collection = new Collection([$transfer]);
|
||||
// $false = new Configuration;
|
||||
// $false->data = false;
|
||||
// FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
// FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
//
|
||||
// // mock stuff
|
||||
// $accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
//
|
||||
// // return single tranfer
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::TRANSFER]])
|
||||
// ->andReturn($collection);
|
||||
//
|
||||
// // for the "other journals" check, return nothing.
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
// ->andReturn(new Collection);
|
||||
//
|
||||
// $accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturn($euro->id);
|
||||
// $currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturn($euro);
|
||||
//
|
||||
// // transaction would be verified, nothing more.
|
||||
// $this->artisan('firefly-iii:journal-currencies')
|
||||
// ->expectsOutput(
|
||||
// sprintf(
|
||||
// 'Transaction #%d has a currency setting #%d that should be #%d. Amount remains %s, currency is changed.',
|
||||
// $source->id,
|
||||
// $source->transaction_currency_id,
|
||||
// $euro->id,
|
||||
// $source->amount
|
||||
// )
|
||||
// )
|
||||
// ->expectsOutput('Verified 2 transaction(s) and journal(s).')
|
||||
// ->assertExitCode(0);
|
||||
// // nothing changed, so no verification.
|
||||
// }
|
||||
//
|
||||
// /**
|
||||
// * Submit a single transfer where the destination account has no currency preference.
|
||||
// *
|
||||
// * @covers \FireflyIII\Console\Commands\Upgrade\JournalCurrencies
|
||||
// */
|
||||
// public function testHandleTransferNoDestinationCurrency(): void
|
||||
// {
|
||||
// // mock classes
|
||||
// $accountRepos = $this->mock(AccountRepositoryInterface::class);
|
||||
// $currencyRepos = $this->mock(CurrencyRepositoryInterface::class);
|
||||
// $journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
// $euro = TransactionCurrency::find(1);
|
||||
// $transfer = $this->getRandomTransfer();
|
||||
//
|
||||
// /** @var Account $destination */
|
||||
// $destination = $transfer->transactions()->where('amount', '>', 0)->first()->account;
|
||||
//
|
||||
// // update transfer if necessary for the test:
|
||||
// $collection = new Collection([$transfer]);
|
||||
// $false = new Configuration;
|
||||
// $false->data = false;
|
||||
// FireflyConfig::shouldReceive('get')->withArgs(['4780_journal_currencies', false])->andReturn($false);
|
||||
// FireflyConfig::shouldReceive('set')->withArgs(['4780_journal_currencies', true]);
|
||||
//
|
||||
// // mock stuff
|
||||
// $accountRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
// $currencyRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
//
|
||||
// // return single tranfer
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::TRANSFER]])
|
||||
// ->andReturn($collection);
|
||||
//
|
||||
// // for the "other journals" check, return nothing.
|
||||
// $journalRepos->shouldReceive('getAllJournals')->atLeast()->once()
|
||||
// ->withArgs([[TransactionType::WITHDRAWAL, TransactionType::DEPOSIT, TransactionType::OPENING_BALANCE, TransactionType::RECONCILIATION,]])
|
||||
// ->andReturn(new Collection);
|
||||
//
|
||||
// $accountRepos->shouldReceive('getMetaValue')->atLeast()->once()->withArgs([Mockery::any(), 'currency_id'])->andReturn($euro->id, 0);
|
||||
// $currencyRepos->shouldReceive('findNull')->atLeast()->once()->andReturn($euro, null);
|
||||
//
|
||||
// // transaction would be verified, nothing more.
|
||||
// $this->artisan('firefly-iii:journal-currencies')
|
||||
// ->expectsOutput(sprintf('Account #%d ("%s") must have currency preference but has none.', $destination->id, $destination->name))
|
||||
// ->assertExitCode(0);
|
||||
// // nothing changed, so no verification.
|
||||
// }
|
||||
|
||||
|
||||
}
|
@ -0,0 +1,96 @@
|
||||
<?php
|
||||
/**
|
||||
* MigrateAttachmentsTest.php
|
||||
* Copyright (c) 2019 thegrumpydictator@gmail.com
|
||||
*
|
||||
* This file is part of Firefly III.
|
||||
*
|
||||
* Firefly III is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Firefly III 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 General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Firefly III. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
namespace Tests\Unit\Console\Commands\Upgrade;
|
||||
|
||||
|
||||
use FireflyConfig;
|
||||
use FireflyIII\Models\Attachment;
|
||||
use FireflyIII\Models\Configuration;
|
||||
use FireflyIII\Models\Note;
|
||||
use FireflyIII\Models\TransactionJournal;
|
||||
use Log;
|
||||
use Tests\TestCase;
|
||||
|
||||
/**
|
||||
* Class MigrateAttachmentsTest
|
||||
*/
|
||||
class MigrateAttachmentsTest extends TestCase
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function setUp(): void
|
||||
{
|
||||
parent::setUp();
|
||||
Log::info(sprintf('Now in %s.', get_class($this)));
|
||||
}
|
||||
|
||||
/**
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateAttachments
|
||||
*/
|
||||
public function testHandle(): void
|
||||
{
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_migrate_attachments', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_migrate_attachments', true]);
|
||||
// assume all is well.
|
||||
$this->artisan('firefly-iii:migrate-attachments')
|
||||
->expectsOutput('All attachments are OK.')
|
||||
->assertExitCode(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateAttachments
|
||||
*/
|
||||
public function testHandleMigrate(): void
|
||||
{
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_migrate_attachments', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_migrate_attachments', true]);
|
||||
|
||||
$attachment = Attachment::create(
|
||||
[
|
||||
'user_id' => 1,
|
||||
'attachable_id' => 1,
|
||||
'attachable_type' => TransactionJournal::class,
|
||||
'description' => 'Hello',
|
||||
'md5' => md5('hello'),
|
||||
'filename' => 'test.pdf',
|
||||
'mime' => 'text/plain',
|
||||
'size' => 1,
|
||||
]);
|
||||
|
||||
// assume all is well.
|
||||
$this->artisan('firefly-iii:migrate-attachments')
|
||||
->expectsOutput('Updated 1 attachment(s).')
|
||||
->assertExitCode(0);
|
||||
|
||||
$this->assertCount(0, Attachment::where('id', $attachment->id)->where('description', '!=', '')->get());
|
||||
$this->assertCount(1, Attachment::where('id', $attachment->id)->where('description', '=', '')->get());
|
||||
$this->assertCount(1, Note::where('noteable_id', $attachment->id)->where('noteable_type', Attachment::class)->get());
|
||||
|
||||
$attachment->forceDelete();
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,94 @@
|
||||
<?php
|
||||
/**
|
||||
* MigrateJournalNotesTest.php
|
||||
* Copyright (c) 2019 thegrumpydictator@gmail.com
|
||||
*
|
||||
* This file is part of Firefly III.
|
||||
*
|
||||
* Firefly III is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Firefly III 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 General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Firefly III. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
namespace Tests\Unit\Console\Commands\Upgrade;
|
||||
|
||||
|
||||
use FireflyConfig;
|
||||
use FireflyIII\Models\Configuration;
|
||||
use FireflyIII\Models\TransactionJournalMeta;
|
||||
use Log;
|
||||
use Tests\TestCase;
|
||||
|
||||
/**
|
||||
* Class MigrateJournalNotesTest
|
||||
*/
|
||||
class MigrateJournalNotesTest extends TestCase
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function setUp(): void
|
||||
{
|
||||
parent::setUp();
|
||||
Log::info(sprintf('Now in %s.', get_class($this)));
|
||||
}
|
||||
|
||||
/**
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateJournalNotes
|
||||
*/
|
||||
public function testHandle(): void
|
||||
{
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_migrate_notes', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_migrate_notes', true]);
|
||||
|
||||
// assume all is well.
|
||||
$this->artisan('firefly-iii:migrate-notes')
|
||||
->expectsOutput('No notes to migrate.')
|
||||
->assertExitCode(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateJournalNotes
|
||||
*/
|
||||
public function testHandleNote(): void
|
||||
{
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_migrate_notes', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_migrate_notes', true]);
|
||||
|
||||
$journal = $this->getRandomWithdrawal();
|
||||
$meta = TransactionJournalMeta::create(
|
||||
[
|
||||
'transaction_journal_id' => $journal->id,
|
||||
'name' => 'notes',
|
||||
'data' => json_encode('Some note.'),
|
||||
'hash' => 'Some hash',
|
||||
]
|
||||
);
|
||||
// assume one is fixed.
|
||||
$this->artisan('firefly-iii:migrate-notes')
|
||||
->expectsOutput('Migrated 1 note(s).')
|
||||
->assertExitCode(0);
|
||||
|
||||
$this->assertCount(0, TransactionJournalMeta
|
||||
::where('name', 'notes')
|
||||
->where('id', $meta->id)
|
||||
->whereNull('deleted_at')
|
||||
->get());
|
||||
|
||||
}
|
||||
|
||||
|
||||
}
|
@ -23,12 +23,13 @@ declare(strict_types=1);
|
||||
|
||||
namespace Tests\Unit\Console\Commands\Upgrade;
|
||||
|
||||
use Carbon\Carbon;
|
||||
use FireflyConfig;
|
||||
use FireflyIII\Factory\TransactionJournalFactory;
|
||||
use FireflyIII\Factory\TransactionGroupFactory;
|
||||
use FireflyIII\Models\Configuration;
|
||||
use FireflyIII\Models\Transaction;
|
||||
use FireflyIII\Models\TransactionJournal;
|
||||
use FireflyIII\Repositories\Journal\JournalRepositoryInterface;
|
||||
use FireflyIII\Services\Internal\Destroy\JournalDestroyService;
|
||||
use Illuminate\Support\Collection;
|
||||
use Log;
|
||||
use Mockery;
|
||||
@ -49,140 +50,199 @@ class MigrateToGroupsTest extends TestCase
|
||||
}
|
||||
|
||||
/**
|
||||
* Basic test. Assume nothing is wrong.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateToGroups
|
||||
*/
|
||||
public function testAlreadyExecuted(): void
|
||||
public function testHandle(): void
|
||||
{
|
||||
$this->markTestIncomplete('Needs to be rewritten for v4.8.0');
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
|
||||
return;
|
||||
$this->mock(TransactionJournalFactory::class);
|
||||
$this->mock(JournalRepositoryInterface::class);
|
||||
// mock calls:
|
||||
$journalRepos->shouldReceive('getSplitJournals')
|
||||
->atLeast()->once()
|
||||
->andReturn(new Collection);
|
||||
$journalRepos->shouldReceive('getJournalsWithoutGroup')
|
||||
->atLeast()->once()
|
||||
->andReturn([]);
|
||||
|
||||
$configObject = new Configuration;
|
||||
$configObject->data = true;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['migrated_to_groups_4780', false])->andReturn($configObject)->once();
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_migrated_to_groups', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_migrated_to_groups', true]);
|
||||
|
||||
// assume all is well.
|
||||
$this->artisan('firefly-iii:migrate-to-groups')
|
||||
->expectsOutput('Database already seems to be migrated.')
|
||||
->expectsOutput('No journals to migrate to groups.')
|
||||
->assertExitCode(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* Return a journal without a group.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateToGroups
|
||||
*/
|
||||
public function testBasic(): void
|
||||
public function testHandleNoGroup(): void
|
||||
{
|
||||
$this->markTestIncomplete('Needs to be rewritten for v4.8.0');
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$asset = $this->getRandomAsset();
|
||||
$expense = $this->getRandomExpense();
|
||||
$journal = TransactionJournal::create(
|
||||
[
|
||||
'user_id' => 1,
|
||||
'transaction_currency_id' => 1,
|
||||
'transaction_type_id' => 1,
|
||||
'description' => 'Test',
|
||||
'tag_count' => 0,
|
||||
'date' => '2019-01-01',
|
||||
]
|
||||
);
|
||||
$one = Transaction::create(
|
||||
[
|
||||
'transaction_journal_id' => $journal->id,
|
||||
'account_id' => $asset->id,
|
||||
'amount' => '-10',
|
||||
'identifier' => 1,
|
||||
]
|
||||
);
|
||||
$two = Transaction::create(
|
||||
[
|
||||
'transaction_journal_id' => $journal->id,
|
||||
'account_id' => $expense->id,
|
||||
'amount' => '10',
|
||||
'identifier' => 1,
|
||||
]
|
||||
);
|
||||
$array = $journal->toArray();
|
||||
|
||||
return;
|
||||
$journalFactory = $this->mock(TransactionJournalFactory::class);
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$withdrawal = $this->getRandomSplitWithdrawal();
|
||||
$collection = new Collection([$withdrawal]);
|
||||
$date = new Carbon;
|
||||
$opposing = new Transaction;
|
||||
$opposing->account_id = 13;
|
||||
// mock calls:
|
||||
$journalRepos->shouldReceive('getSplitJournals')
|
||||
->atLeast()->once()
|
||||
->andReturn(new Collection);
|
||||
$journalRepos->shouldReceive('getJournalsWithoutGroup')
|
||||
->atLeast()->once()
|
||||
->andReturn([$array]);
|
||||
|
||||
// not yet executed:
|
||||
$configObject = new Configuration;
|
||||
$configObject->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['migrated_to_groups_4780', false])->andReturn($configObject)->once();
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['migrated_to_groups_4780', true])->once();
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_migrated_to_groups', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_migrated_to_groups', true]);
|
||||
|
||||
// assume all is well.
|
||||
$this->artisan('firefly-iii:migrate-to-groups')
|
||||
->expectsOutput('Migrated 1 transaction journal(s).')
|
||||
->assertExitCode(0);
|
||||
|
||||
// no longer without a group.
|
||||
$this->assertCount(0, TransactionJournal::where('id', $journal->id)->whereNull('transaction_group_id')->get());
|
||||
$journal->transactionGroup()->forceDelete();
|
||||
$one->forceDelete();
|
||||
$two->forceDelete();
|
||||
$journal->forceDelete();
|
||||
}
|
||||
|
||||
/**
|
||||
* Create split withdrawal and see what the system will do.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateToGroups
|
||||
*/
|
||||
public function testHandleSplitJournal(): void
|
||||
{
|
||||
$asset = $this->getRandomAsset();
|
||||
$expense = $this->getRandomExpense();
|
||||
$group = $this->getRandomWithdrawalGroup();
|
||||
$journal = TransactionJournal::create(
|
||||
[
|
||||
'user_id' => 1,
|
||||
'transaction_currency_id' => 1,
|
||||
'transaction_type_id' => 1,
|
||||
'description' => 'Test',
|
||||
'tag_count' => 0,
|
||||
'date' => '2019-01-01',
|
||||
]
|
||||
);
|
||||
$one = Transaction::create(
|
||||
[
|
||||
'transaction_journal_id' => $journal->id,
|
||||
'account_id' => $asset->id,
|
||||
'amount' => '-10',
|
||||
'identifier' => 1,
|
||||
]
|
||||
);
|
||||
$two = Transaction::create(
|
||||
[
|
||||
'transaction_journal_id' => $journal->id,
|
||||
'account_id' => $expense->id,
|
||||
'amount' => '10',
|
||||
'identifier' => 1,
|
||||
]
|
||||
);
|
||||
$three = Transaction::create(
|
||||
[
|
||||
'transaction_journal_id' => $journal->id,
|
||||
'account_id' => $asset->id,
|
||||
'amount' => '-12',
|
||||
'identifier' => 2,
|
||||
]
|
||||
);
|
||||
$four = Transaction::create(
|
||||
[
|
||||
'transaction_journal_id' => $journal->id,
|
||||
'account_id' => $expense->id,
|
||||
'amount' => '12',
|
||||
'identifier' => 2,
|
||||
]
|
||||
);
|
||||
|
||||
|
||||
// calls to repository:
|
||||
$journalRepos = $this->mock(JournalRepositoryInterface::class);
|
||||
$service = $this->mock(JournalDestroyService::class);
|
||||
$factory = $this->mock(TransactionGroupFactory::class);
|
||||
|
||||
// mock calls:
|
||||
$journalRepos->shouldReceive('setUser')->atLeast()->once();
|
||||
$journalRepos->shouldReceive('getJournalBudgetId')->atLeast()->once()->andReturn(1);
|
||||
$journalRepos->shouldReceive('getJournalCategoryId')->atLeast()->once()->andReturn(2);
|
||||
$journalRepos->shouldReceive('findOpposingTransaction')->atLeast()->once()->andReturn($opposing);
|
||||
$journalRepos->shouldReceive('getNoteText')->atLeast()->once()->andReturn('I am some notes.');
|
||||
$journalRepos->shouldReceive('getTags')->atLeast()->once()->andReturn(['a', 'b']);
|
||||
$journalRepos->shouldReceive('getSplitJournals')->once()->andReturn($collection);
|
||||
|
||||
// all meta field calls.
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'internal-reference'])->andReturn('ABC');
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'sepa-cc'])->andReturnNull();
|
||||
// mock journal things:
|
||||
$journalRepos->shouldReceive('getJournalBudgetId')->atLeast()->once()->andReturn(0);
|
||||
$journalRepos->shouldReceive('getJournalCategoryId')->atLeast()->once()->andReturn(0);
|
||||
$journalRepos->shouldReceive('getNoteText')->atLeast()->once()->andReturn('Some note.');
|
||||
$journalRepos->shouldReceive('getTags')->atLeast()->once()->andReturn(['A', 'B']);
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()
|
||||
->withArgs([Mockery::any(), Mockery::any()])
|
||||
->once()->andReturn(null);
|
||||
$journalRepos->shouldReceive('getMetaDate')->atLeast()
|
||||
->withArgs([Mockery::any(), Mockery::any()])
|
||||
->once()->andReturn(null);
|
||||
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'sepa-ct-op'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'sepa-ct-id'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'sepa-db'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'sepa-country'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'sepa-ep'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'sepa-ci'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'sepa-batch-id'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'external-id'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'original-source'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'recurrence_id'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'bunq_payment_id'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'importHash'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaField')->atLeast()->once()->withArgs([Mockery::any(), 'importHashV2'])->andReturnNull();
|
||||
// create a group
|
||||
$factory->shouldReceive('create')->atLeast()->once()->andReturn($group);
|
||||
$service->shouldReceive('destroy')->atLeast()->once();
|
||||
|
||||
$journalRepos->shouldReceive('getMetaDate')->atLeast()->once()->withArgs([Mockery::any(), 'interest_date'])->andReturn($date);
|
||||
$journalRepos->shouldReceive('getMetaDate')->atLeast()->once()->withArgs([Mockery::any(), 'book_date'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaDate')->atLeast()->once()->withArgs([Mockery::any(), 'process_date'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaDate')->atLeast()->once()->withArgs([Mockery::any(), 'due_date'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaDate')->atLeast()->once()->withArgs([Mockery::any(), 'payment_date'])->andReturnNull();
|
||||
$journalRepos->shouldReceive('getMetaDate')->atLeast()->once()->withArgs([Mockery::any(), 'invoice_date'])->andReturnNull();
|
||||
$factory->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
// calls to factory
|
||||
$journalFactory->shouldReceive('setUser')->atLeast()->once();
|
||||
$journalFactory->shouldReceive('create')->atLeast()->once()->withAnyArgs()->andReturn(new Collection());
|
||||
$journalRepos->shouldReceive('getSplitJournals')
|
||||
->atLeast()->once()
|
||||
->andReturn(new Collection([$journal]));
|
||||
$journalRepos->shouldReceive('getJournalsWithoutGroup')
|
||||
->atLeast()->once()
|
||||
->andReturn([]);
|
||||
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_migrated_to_groups', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_migrated_to_groups', true]);
|
||||
|
||||
$this->artisan('firefly-iii:migrate-to-groups')
|
||||
->expectsOutput('Going to un-split 1 transaction(s). This could take some time.')
|
||||
->expectsOutput('Migrated 1 transaction journal(s).')
|
||||
->assertExitCode(0);
|
||||
|
||||
// delete the created stuff:
|
||||
$one->forceDelete();
|
||||
$two->forceDelete();
|
||||
$three->forceDelete();
|
||||
$four->forceDelete();
|
||||
$journal->forceDelete();
|
||||
|
||||
// the calls above let me know it's OK.
|
||||
}
|
||||
|
||||
/**
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateToGroups
|
||||
*/
|
||||
public function testForced(): void
|
||||
{
|
||||
$this->markTestIncomplete('Needs to be rewritten for v4.8.0');
|
||||
|
||||
return;
|
||||
$this->mock(TransactionJournalFactory::class);
|
||||
$repository = $this->mock(JournalRepositoryInterface::class);
|
||||
|
||||
$repository->shouldReceive('getSplitJournals')->andReturn(new Collection);
|
||||
|
||||
|
||||
$configObject = new Configuration;
|
||||
$configObject->data = true;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['migrated_to_groups_4780', false])->andReturn($configObject)->once();
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['migrated_to_groups_4780', true])->once();
|
||||
|
||||
$this->artisan('firefly-iii:migrate-to-groups --force')
|
||||
->expectsOutput('Forcing the migration.')
|
||||
->expectsOutput('Found no split journals. Nothing to do.')
|
||||
->assertExitCode(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateToGroups
|
||||
*/
|
||||
public function testNotSplit(): void
|
||||
{
|
||||
$this->markTestIncomplete('Needs to be rewritten for v4.8.0');
|
||||
|
||||
return;
|
||||
$this->mock(TransactionJournalFactory::class);
|
||||
$repository = $this->mock(JournalRepositoryInterface::class);
|
||||
$withdrawal = $this->getRandomWithdrawal();
|
||||
|
||||
$repository->shouldReceive('getSplitJournals')->andReturn(new Collection([$withdrawal]));
|
||||
|
||||
|
||||
$configObject = new Configuration;
|
||||
$configObject->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['migrated_to_groups_4780', false])->andReturn($configObject)->once();
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['migrated_to_groups_4780', true])->once();
|
||||
|
||||
$this->artisan('firefly-iii:migrate-to-groups')
|
||||
->expectsOutput('Going to un-split 1 transaction(s). This could take some time.')
|
||||
->assertExitCode(0);
|
||||
}
|
||||
|
||||
}
|
316
tests/Unit/Console/Commands/Upgrade/MigrateToRulesTest.php
Normal file
316
tests/Unit/Console/Commands/Upgrade/MigrateToRulesTest.php
Normal file
@ -0,0 +1,316 @@
|
||||
<?php
|
||||
/**
|
||||
* MigrateToRulesTest.php
|
||||
* Copyright (c) 2019 thegrumpydictator@gmail.com
|
||||
*
|
||||
* This file is part of Firefly III.
|
||||
*
|
||||
* Firefly III is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Firefly III 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 General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with Firefly III. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
namespace Tests\Unit\Console\Commands\Upgrade;
|
||||
|
||||
|
||||
use FireflyConfig;
|
||||
use FireflyIII\Models\Bill;
|
||||
use FireflyIII\Models\Configuration;
|
||||
use FireflyIII\Models\Preference;
|
||||
use FireflyIII\Repositories\Bill\BillRepositoryInterface;
|
||||
use FireflyIII\Repositories\Rule\RuleRepositoryInterface;
|
||||
use FireflyIII\Repositories\RuleGroup\RuleGroupRepositoryInterface;
|
||||
use FireflyIII\Repositories\User\UserRepositoryInterface;
|
||||
use Illuminate\Support\Collection;
|
||||
use Log;
|
||||
use Mockery;
|
||||
use Preferences;
|
||||
use Tests\TestCase;
|
||||
|
||||
/**
|
||||
* Class MigrateToRulesTest
|
||||
*/
|
||||
class MigrateToRulesTest extends TestCase
|
||||
{
|
||||
/**
|
||||
*
|
||||
*/
|
||||
public function setUp(): void
|
||||
{
|
||||
parent::setUp();
|
||||
Log::info(sprintf('Now in %s.', get_class($this)));
|
||||
}
|
||||
|
||||
/**
|
||||
* Basic test. Assume nothing is wrong.
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateToRules
|
||||
*/
|
||||
public function testHandle(): void
|
||||
{
|
||||
|
||||
// mock repositories:
|
||||
$userRepository = $this->mock(UserRepositoryInterface::class);
|
||||
$ruleGroupRepository = $this->mock(RuleGroupRepositoryInterface::class);
|
||||
$billRepository = $this->mock(BillRepositoryInterface::class);
|
||||
$ruleRepository = $this->mock(RuleRepositoryInterface::class);
|
||||
$group = $this->user()->ruleGroups()->inRandomOrder()->first();
|
||||
$bill = $this->user()->bills()->inRandomOrder()->first();
|
||||
// mock all calls.
|
||||
$userRepository->shouldReceive('all')->atLeast()->once()
|
||||
->andReturn(new Collection([$this->user()]));
|
||||
$ruleRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
$ruleGroupRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
$ruleGroupRepository->shouldReceive('findByTitle')
|
||||
->withArgs(['Rule group for bills'])->atLeast()->once()->andReturnNull();
|
||||
// rule group repos should try to store a rule group in response to the result above.
|
||||
$ruleGroupRepository->shouldReceive('store')->atLeast()->once()->andReturn($group);
|
||||
|
||||
// bill repos should return one rule.
|
||||
$billRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
$billRepository->shouldReceive('getBills')->atLeast()->once()
|
||||
->andReturn(new Collection([$bill]));
|
||||
|
||||
// configuration
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_bills_to_rules', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_bills_to_rules', true]);
|
||||
|
||||
// preferences
|
||||
$language = new Preference;
|
||||
$language->data = 'en_US';
|
||||
Preferences::shouldReceive('getForUser')->withArgs([Mockery::any(), 'language', 'en_US'])->andReturn($language);
|
||||
|
||||
|
||||
// assume all is well.
|
||||
$this->artisan('firefly-iii:bills-to-rules')
|
||||
->expectsOutput('All bills are OK.')
|
||||
->assertExitCode(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* Basic test. Give command an unmigrated bill. This bill has the same amount_min
|
||||
* as amount_max
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateToRules
|
||||
*/
|
||||
public function testHandleEvenBill(): void
|
||||
{
|
||||
$bill = Bill::create(
|
||||
[
|
||||
'user_id' => $this->user()->id,
|
||||
'transaction_currency_id' => null,
|
||||
'name' => 'I am a bill',
|
||||
'match' => 'some,kind,of,match',
|
||||
'amount_min' => '30',
|
||||
'amount_max' => '30',
|
||||
'date' => '2019-01-01',
|
||||
'repeat_freq' => 'monthly',
|
||||
]
|
||||
);
|
||||
|
||||
// mock repositories:
|
||||
$userRepository = $this->mock(UserRepositoryInterface::class);
|
||||
$ruleGroupRepository = $this->mock(RuleGroupRepositoryInterface::class);
|
||||
$billRepository = $this->mock(BillRepositoryInterface::class);
|
||||
$ruleRepository = $this->mock(RuleRepositoryInterface::class);
|
||||
$group = $this->user()->ruleGroups()->inRandomOrder()->first();
|
||||
// mock all calls.
|
||||
$userRepository->shouldReceive('all')->atLeast()->once()
|
||||
->andReturn(new Collection([$this->user()]));
|
||||
$ruleRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
// this is what rule repos should receive:
|
||||
$argumentRule = [
|
||||
'rule_group_id' => $group->id,
|
||||
'active' => true,
|
||||
'strict' => false,
|
||||
'stop_processing' => false, // field is no longer used.
|
||||
'title' => 'Auto-generated rule for bill "I am a bill"',
|
||||
'description' => 'This rule is auto-generated to try to match bill "I am a bill".',
|
||||
'trigger' => 'store-journal',
|
||||
'triggers' => [
|
||||
[
|
||||
'type' => 'description_contains',
|
||||
'value' => 'some kind of match',
|
||||
],
|
||||
[
|
||||
'type' => 'amount_exactly',
|
||||
'value' => $bill->amount_min,
|
||||
],
|
||||
],
|
||||
'actions' => [
|
||||
'type' => 'link_to_bill',
|
||||
'value' => $bill->name,
|
||||
],
|
||||
];
|
||||
|
||||
// this is what the bill repos should receive:
|
||||
$argumentBill = [
|
||||
'currency_id' => $bill->transaction_currency_id,
|
||||
'name' => $bill->name,
|
||||
'match' => 'MIGRATED_TO_RULES',
|
||||
'amount_min' => $bill->amount_min,
|
||||
'amount_max' => $bill->amount_max,
|
||||
'date' => $bill->date,
|
||||
'repeat_freq' => $bill->repeat_freq,
|
||||
'skip' => $bill->skip,
|
||||
'active' => $bill->active,
|
||||
];
|
||||
|
||||
|
||||
$ruleRepository->shouldReceive('store')->atLeast()->once()->withArgs([$argumentRule]);
|
||||
|
||||
// rule group repos should try to store a rule group in response to the result above.
|
||||
$ruleGroupRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
$ruleGroupRepository->shouldReceive('findByTitle')
|
||||
->withArgs(['Rule group for bills'])->atLeast()->once()->andReturnNull();
|
||||
|
||||
$ruleGroupRepository->shouldReceive('store')->atLeast()->once()->andReturn($group);
|
||||
|
||||
// bill repos should return one rule.
|
||||
$billRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
$billRepository->shouldReceive('getBills')->atLeast()->once()
|
||||
->andReturn(new Collection([$bill]));
|
||||
$billRepository->shouldReceive('update')->atLeast()->once()
|
||||
->withArgs([Mockery::any(), $argumentBill]);
|
||||
|
||||
// configuration
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_bills_to_rules', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_bills_to_rules', true]);
|
||||
|
||||
// preferences
|
||||
$language = new Preference;
|
||||
$language->data = 'en_US';
|
||||
Preferences::shouldReceive('getForUser')->withArgs([Mockery::any(), 'language', 'en_US'])->andReturn($language);
|
||||
|
||||
|
||||
// assume all is well.
|
||||
$this->artisan('firefly-iii:bills-to-rules')
|
||||
->expectsOutput('Verified and fixed 1 bill(s).')
|
||||
->assertExitCode(0);
|
||||
}
|
||||
|
||||
/**
|
||||
* Basic test. Give command an unmigrated bill. This bill has a different amount_min
|
||||
* from the amount_max
|
||||
*
|
||||
* @covers \FireflyIII\Console\Commands\Upgrade\MigrateToRules
|
||||
*/
|
||||
public function testHandleUnevenBill(): void
|
||||
{
|
||||
$bill = Bill::create(
|
||||
[
|
||||
'user_id' => $this->user()->id,
|
||||
'transaction_currency_id' => null,
|
||||
'name' => 'I am a bill',
|
||||
'match' => 'some,kind,of,match',
|
||||
'amount_min' => '30',
|
||||
'amount_max' => '40',
|
||||
'date' => '2019-01-01',
|
||||
'repeat_freq' => 'monthly',
|
||||
]
|
||||
);
|
||||
|
||||
// mock repositories:
|
||||
$userRepository = $this->mock(UserRepositoryInterface::class);
|
||||
$ruleGroupRepository = $this->mock(RuleGroupRepositoryInterface::class);
|
||||
$billRepository = $this->mock(BillRepositoryInterface::class);
|
||||
$ruleRepository = $this->mock(RuleRepositoryInterface::class);
|
||||
$group = $this->user()->ruleGroups()->inRandomOrder()->first();
|
||||
// mock all calls.
|
||||
$userRepository->shouldReceive('all')->atLeast()->once()
|
||||
->andReturn(new Collection([$this->user()]));
|
||||
$ruleRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
|
||||
// this is what rule repos should receive:
|
||||
$argumentRule = [
|
||||
'rule_group_id' => $group->id,
|
||||
'active' => true,
|
||||
'strict' => false,
|
||||
'stop_processing' => false, // field is no longer used.
|
||||
'title' => 'Auto-generated rule for bill "I am a bill"',
|
||||
'description' => 'This rule is auto-generated to try to match bill "I am a bill".',
|
||||
'trigger' => 'store-journal',
|
||||
'triggers' => [
|
||||
[
|
||||
'type' => 'description_contains',
|
||||
'value' => 'some kind of match',
|
||||
],
|
||||
[
|
||||
'type' => 'amount_less',
|
||||
'value' => $bill->amount_max,
|
||||
],
|
||||
[
|
||||
'type' => 'amount_more',
|
||||
'value' => $bill->amount_min,
|
||||
],
|
||||
],
|
||||
'actions' => [
|
||||
'type' => 'link_to_bill',
|
||||
'value' => $bill->name,
|
||||
],
|
||||
];
|
||||
|
||||
// this is what the bill repos should receive:
|
||||
$argumentBill = [
|
||||
'currency_id' => $bill->transaction_currency_id,
|
||||
'name' => $bill->name,
|
||||
'match' => 'MIGRATED_TO_RULES',
|
||||
'amount_min' => $bill->amount_min,
|
||||
'amount_max' => $bill->amount_max,
|
||||
'date' => $bill->date,
|
||||
'repeat_freq' => $bill->repeat_freq,
|
||||
'skip' => $bill->skip,
|
||||
'active' => $bill->active,
|
||||
];
|
||||
|
||||
|
||||
$ruleRepository->shouldReceive('store')->atLeast()->once()->withArgs([$argumentRule]);
|
||||
|
||||
// rule group repos should try to store a rule group in response to the result above.
|
||||
$ruleGroupRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
$ruleGroupRepository->shouldReceive('findByTitle')
|
||||
->withArgs(['Rule group for bills'])->atLeast()->once()->andReturnNull();
|
||||
|
||||
$ruleGroupRepository->shouldReceive('store')->atLeast()->once()->andReturn($group);
|
||||
|
||||
// bill repos should return one rule.
|
||||
$billRepository->shouldReceive('setUser')->atLeast()->once();
|
||||
$billRepository->shouldReceive('getBills')->atLeast()->once()
|
||||
->andReturn(new Collection([$bill]));
|
||||
$billRepository->shouldReceive('update')->atLeast()->once()
|
||||
->withArgs([Mockery::any(), $argumentBill]);
|
||||
|
||||
// configuration
|
||||
$false = new Configuration;
|
||||
$false->data = false;
|
||||
FireflyConfig::shouldReceive('get')->withArgs(['4780_bills_to_rules', false])->andReturn($false);
|
||||
FireflyConfig::shouldReceive('set')->withArgs(['4780_bills_to_rules', true]);
|
||||
|
||||
// preferences
|
||||
$language = new Preference;
|
||||
$language->data = 'en_US';
|
||||
Preferences::shouldReceive('getForUser')->withArgs([Mockery::any(), 'language', 'en_US'])->andReturn($language);
|
||||
|
||||
|
||||
// assume all is well.
|
||||
$this->artisan('firefly-iii:bills-to-rules')
|
||||
->expectsOutput('Verified and fixed 1 bill(s).')
|
||||
->assertExitCode(0);
|
||||
}
|
||||
|
||||
}
|
Loading…
Reference in New Issue
Block a user