. */ declare(strict_types=1); namespace Tests\Api\V1\Controllers; use FireflyIII\Exceptions\FireflyException; use FireflyIII\Helpers\Collector\TransactionCollector; use FireflyIII\Helpers\Collector\TransactionCollectorInterface; use FireflyIII\Models\Budget; use FireflyIII\Models\BudgetLimit; use FireflyIII\Repositories\Account\AccountRepositoryInterface; use FireflyIII\Repositories\Bill\BillRepositoryInterface; use FireflyIII\Repositories\Budget\BudgetRepositoryInterface; use FireflyIII\Repositories\Currency\CurrencyRepositoryInterface; use FireflyIII\Repositories\Journal\JournalRepositoryInterface; use Laravel\Passport\Passport; use Log; use Tests\TestCase; /** * * Class BudgetLimitControllerTest */ class BudgetLimitControllerTest extends TestCase { /** * */ public function setUp(): void { parent::setUp(); Passport::actingAs($this->user()); Log::info(sprintf('Now in %s.', \get_class($this))); } /** * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController */ public function testDelete(): void { // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); // mock calls: $repository->shouldReceive('setUser')->once(); $repository->shouldReceive('destroyBudgetLimit')->once()->andReturn(true); // Create a budget limit (just in case). /** @var Budget $budget */ $budget = $this->user()->budgets()->first(); $budgetLimit = BudgetLimit::create( [ 'budget_id' => $budget->id, 'start_date' => '2018-01-01', 'end_date' => '2018-01-31', 'amount' => 1, ] ); // call API $response = $this->delete('/api/v1/budgets/limits/' . $budgetLimit->id); $response->assertStatus(204); } /** * Show budget limits by budget, include no dates. * * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController */ public function testIndex(): void { /** @var Budget $budget */ $budget = $this->user()->budgets()->first(); // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); // mock calls: $repository->shouldReceive('setUser')->once(); $repository->shouldReceive('findNull')->andReturn($budget); $repository->shouldReceive('getBudgetLimits')->once()->andReturn($budget->budgetlimits()->get()); // call API $params = [ 'budget_id' => $budget->id, ]; $response = $this->get('/api/v1/budgets/limits?' . http_build_query($params)); $response->assertStatus(200); $response->assertHeader('Content-Type', 'application/vnd.api+json'); } /** * Show budget limits by budget, include dates. * * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController */ public function testIndexNoBudget(): void { /** @var Budget $budget */ $budget = $this->user()->budgets()->first(); // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); // mock calls: $repository->shouldReceive('setUser')->once(); $repository->shouldReceive('findNull')->andReturn(null); $repository->shouldReceive('getAllBudgetLimits')->once()->andReturn($budget->budgetlimits()->get()); // call API $params = [ 'start' => '2018-01-01', 'end' => '2018-01-31', ]; $uri = '/api/v1/budgets/limits?' . http_build_query($params); $response = $this->get($uri); $response->assertStatus(200); $response->assertHeader('Content-Type', 'application/vnd.api+json'); } /** * Show budget limits by budget, include dates. * * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController */ public function testIndexWithDates(): void { /** @var Budget $budget */ $budget = $this->user()->budgets()->first(); // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); // mock calls: $repository->shouldReceive('setUser')->once(); $repository->shouldReceive('findNull')->andReturn($budget); $repository->shouldReceive('getBudgetLimits')->once()->andReturn($budget->budgetlimits()->get()); // call API $params = [ 'budget_id' => $budget->id, 'start' => '2018-01-01', 'end' => '2018-01-31', ]; $response = $this->get('/api/v1/budgets/limits?' . http_build_query($params)); $response->assertStatus(200); $response->assertHeader('Content-Type', 'application/vnd.api+json'); } /** * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController */ public function testShow(): void { // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); // mock calls: $repository->shouldReceive('setUser')->once(); // Create a budget limit (just in case). /** @var Budget $budget */ $budget = $this->user()->budgets()->first(); $budgetLimit = BudgetLimit::create( [ 'budget_id' => $budget->id, 'start_date' => '2018-01-01', 'end_date' => '2018-01-31', 'amount' => 1, ] ); $response = $this->get('/api/v1/budgets/limits/' . $budgetLimit->id); $response->assertStatus(200); $response->assertHeader('Content-Type', 'application/vnd.api+json'); } /** * Store new budget limit. * * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController * @covers \FireflyIII\Api\V1\Requests\BudgetLimitRequest */ public function testStore(): void { $budget = $this->user()->budgets()->first(); $budgetLimit = BudgetLimit::create( [ 'budget_id' => $budget->id, 'start_date' => '2018-01-01', 'end_date' => '2018-01-31', 'amount' => 1, ] ); $data = [ 'budget_id' => $budget->id, 'start' => '2018-01-01', 'end' => '2018-01-31', 'amount' => 1, ]; // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); $repository->shouldReceive('findNull')->andReturn($budget)->once(); $repository->shouldReceive('storeBudgetLimit')->andReturn($budgetLimit)->once(); // mock calls: $repository->shouldReceive('setUser')->once(); // call API $response = $this->post('/api/v1/budgets/limits', $data, ['Accept' => 'application/json']); $response->assertStatus(200); $response->assertHeader('Content-Type', 'application/vnd.api+json'); } /** * Store new budget limit, but give error * * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController * @covers \FireflyIII\Api\V1\Requests\BudgetLimitRequest */ public function testStoreBadBudget(): void { $data = [ 'budget_id' => '1', 'start' => '2018-01-01', 'end' => '2018-01-31', 'amount' => 1, ]; // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); $repository->shouldReceive('findNull')->andReturn(null)->once(); // mock calls: $repository->shouldReceive('setUser')->once(); // call API $response = $this->post('/api/v1/budgets/limits', $data); $response->assertStatus(500); $response->assertSee('Unknown budget.'); } /** * Show index. * * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController */ public function testTransactionsBasic(): void { $budgetLimit = BudgetLimit::first(); // get some transactions using the collector: Log::info('This transaction collector is OK, because it is used in a test:'); $collector = new TransactionCollector; $collector->setUser($this->user()); $collector->withOpposingAccount()->withCategoryInformation()->withBudgetInformation(); $collector->setAllAssetAccounts(); $collector->setLimit(5)->setPage(1); try { $paginator = $collector->getPaginatedTransactions(); } catch (FireflyException $e) { $this->assertTrue(false, $e->getMessage()); } // mock stuff: $repository = $this->mock(JournalRepositoryInterface::class); $collector = $this->mock(TransactionCollectorInterface::class); $currencyRepository = $this->mock(CurrencyRepositoryInterface::class); $accountRepos = $this->mock(AccountRepositoryInterface::class); $billRepos = $this->mock(BillRepositoryInterface::class); $budgetRepos = $this->mock(BudgetRepositoryInterface::class); $billRepos->shouldReceive('setUser'); $repository->shouldReceive('setUser'); $currencyRepository->shouldReceive('setUser'); $budgetRepos->shouldReceive('setUser'); $repository->shouldReceive('getNoteText')->atLeast()->once()->andReturn('Note'); $repository->shouldReceive('getMetaField')->atLeast()->once()->andReturn(null); $repository->shouldReceive('getMetaDateString')->atLeast()->once()->andReturn('2018-01-01'); $collector->shouldReceive('setUser')->andReturnSelf(); $collector->shouldReceive('withOpposingAccount')->andReturnSelf(); $collector->shouldReceive('withCategoryInformation')->andReturnSelf(); $collector->shouldReceive('withBudgetInformation')->andReturnSelf(); $collector->shouldReceive('setBudget')->andReturnSelf(); $collector->shouldReceive('removeFilter')->andReturnSelf(); $collector->shouldReceive('setLimit')->andReturnSelf(); $collector->shouldReceive('setRange')->andReturnSelf(); $collector->shouldReceive('setPage')->andReturnSelf(); $collector->shouldReceive('setTypes')->andReturnSelf(); $collector->shouldReceive('setAllAssetAccounts')->andReturnSelf(); $collector->shouldReceive('getPaginatedTransactions')->andReturn($paginator); // mock some calls: // test API $response = $this->get(route('api.v1.budget_limits.transactions', [$budgetLimit->id])); $response->assertStatus(200); $response->assertJson(['data' => [],]); $response->assertJson(['meta' => ['pagination' => ['total' => true, 'count' => true, 'per_page' => 5, 'current_page' => 1, 'total_pages' => true]],]); $response->assertJson(['links' => ['self' => true, 'first' => true, 'last' => true,],]); $response->assertHeader('Content-Type', 'application/vnd.api+json'); } /** * Test update of budget limit. * * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController * @covers \FireflyIII\Api\V1\Requests\BudgetLimitRequest */ public function testUpdate(): void { $budget = $this->user()->budgets()->first(); $budgetLimit = BudgetLimit::create( [ 'budget_id' => $budget->id, 'start_date' => '2018-01-01', 'end_date' => '2018-01-31', 'amount' => 1, ] ); $data = [ 'budget_id' => $budget->id, 'start' => '2018-01-01', 'end' => '2018-01-31', 'amount' => 2, ]; // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); $repository->shouldReceive('findNull')->andReturn($budget)->once(); $repository->shouldReceive('updateBudgetLimit')->andReturn($budgetLimit)->once(); // mock calls: $repository->shouldReceive('setUser')->once(); // call API $response = $this->put('/api/v1/budgets/limits/' . $budgetLimit->id, $data); $response->assertStatus(200); $response->assertHeader('Content-Type', 'application/vnd.api+json'); } /** * Test update of budget limit but submit bad budget. * * @covers \FireflyIII\Api\V1\Controllers\BudgetLimitController * @covers \FireflyIII\Api\V1\Requests\BudgetLimitRequest */ public function testUpdateBadBudget(): void { $budget = $this->user()->budgets()->first(); $budgetLimit = BudgetLimit::create( [ 'budget_id' => $budget->id, 'start_date' => '2018-01-01', 'end_date' => '2018-01-31', 'amount' => 1, ] ); $data = [ 'budget_id' => $budget->id, 'start' => '2018-01-01', 'end' => '2018-01-31', 'amount' => 2, ]; // mock stuff: $repository = $this->mock(BudgetRepositoryInterface::class); $repository->shouldReceive('findNull')->andReturn(null)->once(); // mock calls: $repository->shouldReceive('setUser')->once(); // call API $response = $this->put('/api/v1/budgets/limits/' . $budgetLimit->id, $data); $response->assertStatus(500); $response->assertSee('Unknown budget.'); } }