feat(project): Complete Project Lifecycle capability with full TDD workflow

- Implement ProjectController with CRUD, status transitions, estimate/forecast
- Add ProjectService with state machine validation
- Extract ProjectStatusService for reusable state machine logic
- Add ProjectPolicy for role-based authorization
- Create ProjectSeeder with test data
- Implement frontend project management UI with modal forms
- Add projectService API client
- Complete all 9 incomplete unit tests (ProjectModelTest, ProjectForecastTest, ProjectPolicyTest)
- Fix E2E test timing issues with loading state waits
- Add Scribe API documentation annotations
- Improve forecasted effort validation messages with detailed feedback

Test Results:
- Backend: 49 passed (182 assertions)
- Frontend Unit: 32 passed
- E2E: 134 passed (Chromium + Firefox)

Phase 3 Refactor:
- Extract ProjectStatusService for state machine
- Optimize project list query with status joins
- Improve forecasted effort validation messages

Phase 4 Document:
- Add Scribe annotations to ProjectController
- Generate API documentation
This commit is contained in:
2026-02-19 02:43:05 -05:00
parent 8f70e81d29
commit 8ed56c9f7c
19 changed files with 5126 additions and 173 deletions

View File

@@ -2,11 +2,15 @@
namespace Tests\Unit\Models;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Tests\TestCase;
use App\Models\Project;
use App\Models\ProjectStatus;
use App\Models\ProjectType;
use App\Services\ProjectService;
use App\Services\ProjectStatusService;
use Database\Seeders\ProjectStatusSeeder;
use Database\Seeders\ProjectTypeSeeder;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Tests\TestCase;
class ProjectModelTest extends TestCase
{
@@ -15,16 +19,48 @@ class ProjectModelTest extends TestCase
// 3.1.22 Unit test: Project status state machine
public function test_project_status_state_machine()
{
$this->markTestIncomplete('3.1.22: Implement project status state machine tests');
$statusService = app(ProjectStatusService::class);
$this->assertContains('SOW Approval', $statusService->getValidTransitions('Pre-sales'));
}
public function test_project_can_transition_to_valid_status()
{
$this->markTestIncomplete('3.1.22: Test valid status transitions');
$this->seed([ProjectStatusSeeder::class, ProjectTypeSeeder::class]);
$service = app(ProjectService::class);
$preSales = ProjectStatus::where('name', 'Pre-sales')->firstOrFail();
$sowApproval = ProjectStatus::where('name', 'SOW Approval')->firstOrFail();
$type = ProjectType::firstOrFail();
$project = Project::factory()->create([
'status_id' => $preSales->id,
'type_id' => $type->id,
]);
$updated = $service->transitionStatus($project, $sowApproval->id);
$this->assertSame($sowApproval->id, $updated->status_id);
$this->assertSame('SOW Approval', $updated->status->name);
}
public function test_project_cannot_transition_to_invalid_status()
{
$this->markTestIncomplete('3.1.22: Test invalid status transitions are rejected');
$this->seed([ProjectStatusSeeder::class, ProjectTypeSeeder::class]);
$service = app(ProjectService::class);
$preSales = ProjectStatus::where('name', 'Pre-sales')->firstOrFail();
$inProgress = ProjectStatus::where('name', 'In Progress')->firstOrFail();
$type = ProjectType::firstOrFail();
$project = Project::factory()->create([
'status_id' => $preSales->id,
'type_id' => $type->id,
]);
$this->expectException(\RuntimeException::class);
$service->transitionStatus($project, $inProgress->id);
}
}