TenantAtlas/apps/platform/tests/Unit/Alerts/AlertRetryPolicyTest.php
ahmido ce0615a9c1 Spec 182: relocate Laravel platform to apps/platform (#213)
## Summary
- move the Laravel application into `apps/platform` and keep the repository root for orchestration, docs, and tooling
- update the local command model, Sail/Docker wiring, runtime paths, and ignore rules around the new platform location
- add relocation quickstart/contracts plus focused smoke coverage for bootstrap, command model, routes, and runtime behavior

## Validation
- `cd apps/platform && ./vendor/bin/sail artisan test --compact tests/Feature/PlatformRelocation`
- integrated browser smoke validated `/up`, `/`, `/admin`, `/admin/choose-workspace`, and tenant route semantics for `200`, `403`, and `404`

## Remaining Rollout Checks
- validate Dokploy build context and working-directory assumptions against the new `apps/platform` layout
- confirm web, queue, and scheduler processes all start from the expected working directory in staging/production
- verify no legacy volume mounts or asset-publish paths still point at the old root-level `public/` or `storage/` locations

Co-authored-by: Ahmed Darrazi <ahmed.darrazi@live.de>
Reviewed-on: #213
2026-04-08 08:40:47 +00:00

71 lines
2.4 KiB
PHP

<?php
declare(strict_types=1);
use App\Jobs\Alerts\DeliverAlertsJob;
use App\Models\AlertDelivery;
use App\Models\AlertDestination;
use App\Models\AlertRule;
use App\Services\Alerts\AlertSender;
use Illuminate\Foundation\Testing\RefreshDatabase;
uses(RefreshDatabase::class);
it('marks delivery as failed after bounded retry attempts', function (): void {
[, $tenant] = createUserWithTenant(role: 'owner');
$workspaceId = (int) $tenant->workspace_id;
$destination = AlertDestination::factory()->create([
'workspace_id' => $workspaceId,
'type' => AlertDestination::TYPE_TEAMS_WEBHOOK,
'config' => ['webhook_url' => 'https://example.invalid/hook'],
]);
$rule = AlertRule::factory()->create([
'workspace_id' => $workspaceId,
]);
$rule->destinations()->syncWithPivotValues([(int) $destination->getKey()], ['workspace_id' => $workspaceId]);
$delivery = AlertDelivery::factory()->create([
'workspace_id' => $workspaceId,
'tenant_id' => (int) $tenant->getKey(),
'alert_rule_id' => (int) $rule->getKey(),
'alert_destination_id' => (int) $destination->getKey(),
'status' => AlertDelivery::STATUS_QUEUED,
'attempt_count' => 0,
'send_after' => null,
]);
$sender = \Mockery::mock(AlertSender::class);
$sender->shouldReceive('send')
->times(3)
->andThrow(new RuntimeException('simulated sender failure'));
app()->instance(AlertSender::class, $sender);
$runJob = static function (int $workspaceId): void {
$job = new DeliverAlertsJob($workspaceId);
app()->call([$job, 'handle']);
};
$runJob($workspaceId);
$delivery->refresh();
expect($delivery->attempt_count)->toBe(1);
expect($delivery->status)->toBe(AlertDelivery::STATUS_QUEUED);
expect($delivery->send_after)->not->toBeNull();
$delivery->forceFill(['send_after' => now()->subSecond()])->save();
$runJob($workspaceId);
$delivery->refresh();
expect($delivery->attempt_count)->toBe(2);
expect($delivery->status)->toBe(AlertDelivery::STATUS_QUEUED);
expect($delivery->send_after)->not->toBeNull();
$delivery->forceFill(['send_after' => now()->subSecond()])->save();
$runJob($workspaceId);
$delivery->refresh();
expect($delivery->attempt_count)->toBe(3);
expect($delivery->status)->toBe(AlertDelivery::STATUS_FAILED);
expect($delivery->last_error_message)->toContain('simulated sender failure');
});