Files
investbrain/tests/SyncDailyChangeTest.php
T

168 lines
7.3 KiB
PHP
Raw Permalink Normal View History

2024-09-11 22:00:37 -05:00
<?php
2025-01-28 17:33:54 -06:00
declare(strict_types=1);
2024-09-11 22:00:37 -05:00
namespace Tests;
2025-01-28 17:14:49 -06:00
use App\Models\DailyChange;
2024-09-11 22:00:37 -05:00
use App\Models\Holding;
use App\Models\Portfolio;
use App\Models\Transaction;
2025-01-28 17:14:49 -06:00
use App\Models\User;
use Carbon\CarbonPeriod;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Illuminate\Support\Carbon;
2024-09-11 22:00:37 -05:00
use Illuminate\Support\Facades\Artisan;
class SyncDailyChangeTest extends TestCase
{
use RefreshDatabase;
public function test_can_sync_daily_change_history(): void
{
$this->actingAs($user = User::factory()->create());
$portfolio = Portfolio::factory()->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('ACME')->create();
Transaction::factory()->sell()->lastMonth()->portfolio($portfolio->id)->symbol('ACME')->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('AAPL')->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('GOOG')->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('FOO')->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('BAR')->create();
$portfolio->syncDailyChanges();
$count_of_daily_changes = $portfolio->daily_change()->count('date');
$days_between_now_and_first_trans = (int) CarbonPeriod::create(
2025-01-28 17:14:49 -06:00
$portfolio->transactions()->min('date'),
now()->isBefore(Carbon::parse(config('investbrain.daily_change_time_of_day'))) ? now()->subDay() : now()
)->filter('isWeekday')
2025-01-28 17:14:49 -06:00
->count();
2024-09-11 22:00:37 -05:00
$this->assertEquals($count_of_daily_changes, $days_between_now_and_first_trans);
}
2024-09-15 11:28:13 -05:00
public function test_cost_basis_is_synced(): void
{
$this->actingAs($user = User::factory()->create());
$portfolio = Portfolio::factory()->create();
$first_transaction = Transaction::factory()->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('ACME')->create();
Artisan::call('sync:daily-change', ['portfolio_id' => $portfolio->id]);
$holding = Holding::symbol('ACME')->portfolio($portfolio->id)->first();
$daily_change = DailyChange::whereDate('date', '<=', $first_transaction->date->addDays(2))
2025-01-28 17:14:49 -06:00
->whereDate('date', '>=', $first_transaction->date->subDays(2))
->orderByDesc('date')
->first();
2024-09-15 11:28:13 -05:00
$this->assertEquals($holding->average_cost_basis, $daily_change->total_cost_basis);
$second_transaction = Transaction::factory()->buy()->lastYear()->portfolio($portfolio->id)->symbol('ACME')->create();
Artisan::call('sync:daily-change', ['portfolio_id' => $portfolio->id]);
$daily_change = DailyChange::whereDate('date', '<=', $second_transaction->date->addDays(2))
2025-01-28 17:14:49 -06:00
->whereDate('date', '>=', $second_transaction->date->subDays(2))
->orderByDesc('date')
->first();
$this->assertEqualsWithDelta($first_transaction->cost_basis + $second_transaction->cost_basis, $daily_change->total_cost_basis, 0.01);
2024-09-15 11:28:13 -05:00
$third_transaction = Transaction::factory(2)->sell()->lastMonth()->portfolio($portfolio->id)->symbol('ACME')->create()->first();
Artisan::call('sync:daily-change', ['portfolio_id' => $portfolio->id]);
$daily_change = DailyChange::whereDate('date', '<=', $third_transaction->date->addDays(2))
2025-01-28 17:14:49 -06:00
->whereDate('date', '>=', $third_transaction->date->subDays(2))
->orderByDesc('date')
->first();
2024-09-15 11:28:13 -05:00
2025-01-28 17:14:49 -06:00
$this->assertEquals(0, $daily_change->total_cost_basis);
2024-09-15 11:28:13 -05:00
}
2024-09-11 22:00:37 -05:00
public function test_sales_are_captured_as_realized_gains(): void
{
$this->actingAs($user = User::factory()->create());
$portfolio = Portfolio::factory()->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('ACME')->create();
$sale_transaction = Transaction::factory()->sell()->lastMonth()->portfolio($portfolio->id)->symbol('ACME')->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('AAPL')->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('GOOG')->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('FOO')->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('BAR')->create();
$portfolio->syncDailyChanges();
$daily_change = DailyChange::query()
->portfolio($portfolio->id)
2024-10-01 13:41:50 -05:00
->whereDate('date', '<=', $sale_transaction->date->addDays(2))
->whereDate('date', '>=', $sale_transaction->date->subDays(2))
->orderByDesc('date')
2024-09-11 22:00:37 -05:00
->first();
$realized_gain = ($sale_transaction->sale_price - $sale_transaction->cost_basis) * $sale_transaction->quantity;
$this->assertEqualsWithDelta($daily_change->realized_gains, $realized_gain, 0.01);
$day_before = DailyChange::query()
->portfolio($portfolio->id)
2024-10-01 13:41:50 -05:00
->whereDate('date', '<', $sale_transaction->date->subDays(1))
->orderByDesc('date')
->limit(10)
2024-09-11 22:00:37 -05:00
->first();
$this->assertEquals($day_before->realized_gains, 0);
$after = DailyChange::query()
2024-09-11 22:00:37 -05:00
->portfolio($portfolio->id)
->whereDate('date', '<=', $sale_transaction->date->addDays(2))
->whereDate('date', '>=', $sale_transaction->date->subDays(2))
->orderByDesc('date')
2024-09-11 22:00:37 -05:00
->first();
$this->assertEqualsWithDelta($after->realized_gains, $realized_gain, 0.01);
2024-09-11 22:00:37 -05:00
}
public function test_dividends_captured_in_daily_change_sync(): void
{
$this->actingAs($user = User::factory()->create());
$portfolio = Portfolio::factory()->create();
Transaction::factory(5)->buy()->yearsAgo()->portfolio($portfolio->id)->symbol('ACME')->create();
Artisan::call('refresh:dividend-data');
$portfolio->syncDailyChanges();
$holding = Holding::query()->portfolio($portfolio->id)->symbol('ACME')->first();
$dividends = $holding->dividends()->get()->sortBy('date');
$first_dividend_change = DailyChange::query()
->portfolio($portfolio->id)
->whereDate('date', '<=', $dividends->first()->date->addDays(2))
->whereDate('date', '>=', $dividends->first()->date->subDays(2))
->orderByDesc('date')
2024-09-11 22:00:37 -05:00
->first();
$owned = $dividends->first()->purchased - $dividends->first()->sold;
$this->assertEqualsWithDelta($dividends->first()->dividend_amount * $owned, $first_dividend_change->total_dividends_earned, 0.01);
$last_dividend_change = DailyChange::query()
->portfolio($portfolio->id)
->whereDate('date', '<=', $dividends->last()->date->addDays(2))
->whereDate('date', '>=', $dividends->last()->date->subDays(2))
->orderByDesc('date')
2024-09-11 22:00:37 -05:00
->first();
$total_dividends = $dividends->reduce(function (?float $carry, $dividend) {
return $carry + ($dividend['dividend_amount'] * ($dividend['purchased'] - $dividend['sold']));
});
$owned = $dividends->last()->purchased - $dividends->last()->sold;
$this->assertEqualsWithDelta($total_dividends, $last_dividend_change->total_dividends_earned, 0.01);
}
}