seed(CurrencySeeder::class); $this->seed(PlayTypeSeeder::class); $this->seed(OperationalConfigV1Seeder::class); $this->seed(LotterySettingsSeeder::class); }); test('settlement pays big winner and marks ticket settled', function (): void { $uniq = bin2hex(random_bytes(4)); $player = Player::query()->create([ 'site_code' => 'test', 'site_player_id' => 'settle-p-'.$uniq, 'username' => 'sp_'.$uniq, 'nickname' => null, 'default_currency' => 'NPR', 'status' => 0, ]); PlayerWallet::query()->create([ 'player_id' => $player->id, 'wallet_type' => 'lottery', 'currency_code' => 'NPR', 'balance' => 5_000_000, 'frozen_balance' => 0, 'status' => 0, 'version' => 0, ]); $draw = Draw::query()->create([ 'draw_no' => '20260511-900', 'business_date' => '2026-05-11', 'sequence_no' => 900, 'status' => DrawStatus::Open->value, 'start_time' => now()->subMinutes(2), 'close_time' => now()->addMinutes(5), 'draw_time' => now()->addMinutes(6), 'cooling_end_time' => null, 'result_source' => null, 'current_result_version' => 0, 'settle_version' => 0, 'is_reopened' => false, ]); $this->withHeader('Authorization', 'Bearer dev:'.$player->id) ->postJson('/api/v1/ticket/place', [ 'draw_id' => '20260511-900', 'currency_code' => 'NPR', 'client_trace_id' => 'settle-trace-1', 'lines' => [ ['number' => '1234', 'play_code' => 'big', 'amount' => 10_000], ], ]) ->assertOk(); $batch = DrawResultBatch::query()->create([ 'draw_id' => $draw->id, 'result_version' => 1, 'source_type' => 'rng', 'rng_seed_hash' => 'test', 'raw_seed_encrypted' => null, 'status' => DrawResultBatchStatus::Published->value, 'created_by' => null, 'confirmed_by' => null, 'confirmed_at' => now(), ]); foreach (DrawPrizeLayout::slots() as $slot) { $num = $slot['prize_type'] === 'first' ? '1234' : '5678'; $suffix3 = substr($num, -3); $suffix2 = substr($num, -2); DrawResultItem::query()->create([ 'draw_id' => $draw->id, 'result_batch_id' => $batch->id, 'prize_type' => $slot['prize_type'], 'prize_index' => $slot['prize_index'], 'number_4d' => $num, 'suffix_3d' => $suffix3, 'suffix_2d' => $suffix2, 'head_digit' => (int) substr($num, 0, 1), 'tail_digit' => (int) substr($num, 3, 1), ]); } $draw->forceFill([ 'status' => DrawStatus::Settling->value, 'current_result_version' => 1, ])->save(); $ran = app(SettlementOrchestrator::class)->trySettleDraw($draw->fresh()); expect($ran)->toBeTrue(); $admin = AdminUser::query()->create([ 'username' => 'settlement_legacy_reviewer', 'name' => 'Settlement Legacy Reviewer', 'email' => null, 'password' => Hash::make('secret-strong'), 'status' => 0, ]); grantSuperAdminRole($admin); $settlementBatch = SettlementBatch::query()->where('draw_id', $draw->id)->firstOrFail(); app(SettlementBatchWorkflowService::class)->approve($settlementBatch, $admin); app(SettlementBatchWorkflowService::class)->payout($settlementBatch->fresh()); $draw->refresh(); expect($draw->status)->toBe(DrawStatus::Settled->value); expect((int) $draw->settle_version)->toBe(1); $item = TicketItem::query()->where('draw_id', $draw->id)->firstOrFail(); expect($item->status)->toBe('settled_win'); expect((int) $item->win_amount)->toBe(250_000); $order = TicketOrder::query()->whereKey($item->order_id)->firstOrFail(); expect($order->status)->toBe('settled'); expect(SettlementBatch::query()->where('draw_id', $draw->id)->count())->toBe(1); $wallet = PlayerWallet::query()->where('player_id', $player->id)->firstOrFail(); expect((int) $wallet->balance)->toBe(5_000_000 - (int) $item->actual_deduct_amount + 250_000); expect(WalletTxn::query()->where('biz_type', 'settle_payout')->count())->toBe(1); }); test('admin settlement requires review before payout and can export report', function (): void { $uniq = bin2hex(random_bytes(4)); $player = Player::query()->create([ 'site_code' => 'test', 'site_player_id' => 'settle-review-p-'.$uniq, 'username' => 'srp_'.$uniq, 'nickname' => null, 'default_currency' => 'NPR', 'status' => 0, ]); PlayerWallet::query()->create([ 'player_id' => $player->id, 'wallet_type' => 'lottery', 'currency_code' => 'NPR', 'balance' => 5_000_000, 'frozen_balance' => 0, 'status' => 0, 'version' => 0, ]); $draw = Draw::query()->create([ 'draw_no' => '20260511-901', 'business_date' => '2026-05-11', 'sequence_no' => 901, 'status' => DrawStatus::Open->value, 'start_time' => now()->subMinutes(2), 'close_time' => now()->addMinutes(5), 'draw_time' => now()->addMinutes(6), 'cooling_end_time' => null, 'result_source' => null, 'current_result_version' => 0, 'settle_version' => 0, 'is_reopened' => false, ]); $this->withHeader('Authorization', 'Bearer dev:'.$player->id) ->postJson('/api/v1/ticket/place', [ 'draw_id' => '20260511-901', 'currency_code' => 'NPR', 'client_trace_id' => 'settle-review-trace-1', 'lines' => [ ['number' => '1234', 'play_code' => 'big', 'amount' => 10_000], ], ]) ->assertOk(); $batch = DrawResultBatch::query()->create([ 'draw_id' => $draw->id, 'result_version' => 1, 'source_type' => 'rng', 'rng_seed_hash' => 'review-test', 'raw_seed_encrypted' => null, 'status' => DrawResultBatchStatus::Published->value, 'created_by' => null, 'confirmed_by' => null, 'confirmed_at' => now(), ]); foreach (DrawPrizeLayout::slots() as $slot) { $num = $slot['prize_type'] === 'first' ? '1234' : '5678'; DrawResultItem::query()->create([ 'draw_id' => $draw->id, 'result_batch_id' => $batch->id, 'prize_type' => $slot['prize_type'], 'prize_index' => $slot['prize_index'], 'number_4d' => $num, 'suffix_3d' => substr($num, -3), 'suffix_2d' => substr($num, -2), 'head_digit' => (int) substr($num, 0, 1), 'tail_digit' => (int) substr($num, 3, 1), ]); } $draw->forceFill([ 'status' => DrawStatus::Settling->value, 'current_result_version' => 1, ])->save(); $admin = AdminUser::query()->create([ 'username' => 'settlement_reviewer', 'name' => 'Settlement Reviewer', 'email' => null, 'password' => Hash::make('secret-strong'), 'status' => 0, ]); grantSuperAdminRole($admin); $token = $admin->createToken('test', ['*'], now()->addDay())->plainTextToken; $this->withHeader('Authorization', 'Bearer '.$token) ->postJson("/api/v1/admin/draws/{$draw->id}/settlement/run") ->assertOk() ->assertJsonPath('data.status', DrawStatus::Settling->value); $settlement = SettlementBatch::query()->where('draw_id', $draw->id)->firstOrFail(); expect($settlement->status)->toBe('pending_review'); $item = TicketItem::query()->where('draw_id', $draw->id)->firstOrFail(); expect($item->status)->toBe('pending_payout'); expect(WalletTxn::query()->where('biz_type', 'settle_payout')->count())->toBe(0); $this->withHeader('Authorization', 'Bearer '.$token) ->postJson("/api/v1/admin/settlement-batches/{$settlement->id}/approve") ->assertOk() ->assertJsonPath('data.review_status', 'approved'); $this->withHeader('Authorization', 'Bearer '.$token) ->postJson("/api/v1/admin/settlement-batches/{$settlement->id}/payout") ->assertOk() ->assertJsonPath('data.status', 'paid'); $this->withHeader('Authorization', 'Bearer '.$token) ->getJson('/api/v1/admin/settlement-batches?draw_no=20260511-901&status=paid') ->assertOk() ->assertJsonPath('data.items.0.total_bet_amount', 10_000) ->assertJsonPath('data.items.0.total_actual_deduct', 10_000) ->assertJsonPath('data.items.0.total_payout_amount', 250_000) ->assertJsonPath('data.items.0.platform_profit', -240_000); $this->withHeader('Authorization', 'Bearer '.$token) ->getJson("/api/v1/admin/settlement-batches/{$settlement->id}") ->assertOk() ->assertJsonPath('data.total_bet_amount', 10_000) ->assertJsonPath('data.total_actual_deduct', 10_000) ->assertJsonPath('data.platform_profit', -240_000); $item->refresh(); expect($item->status)->toBe('settled_win'); expect(WalletTxn::query()->where('biz_type', 'settle_payout')->count())->toBe(1); $this->withHeader('Authorization', 'Bearer '.$token) ->get("/api/v1/admin/settlement-batches/{$settlement->id}/export") ->assertOk() ->assertHeader('content-type', 'text/csv; charset=UTF-8'); });