Files
crewli/api/tests/Feature/FormBuilder/ConditionalLogic/ConditionalLogicCycleDetectionTest.php
bert.hausmans d06ea01b09 feat(form-builder): FormFieldConditionalLogicService + cycle detection + legacy backfill + snapshot
WS-5c commit 2 of 4 — the service layer, backfill migration, and
read-path switch. Per addendum Q3, conditional_logic applies to
FormField only — no library mirror and no copyLogic on
FormFieldService::insertFromLibrary.

FormFieldConditionalLogicService owns every write:
  - logicFor(field): depth-limited eager-load of the tree
  - replaceLogic(field, tree): transactional structure + operator +
    field_slug validation + cycle check + activity-log emit
    (field.conditional_logic_replaced)
  - toJsonShape(root): reconstructs the canonical ARCH §8
    `{show_when: {...}}` shape — single source of truth for the
    snapshot writer + API resources
  - assertSpecsValid(tree): public boundary guard for the FormRequest
    strict validator (WS-5c commit 3 wires this up)
  - assertNoCycles(field, tree): contract preserved from
    FormFieldService::assertNoConditionalCycle, implementation now
    reads the relational adjacency.

Backfill migration translates pre-WS-5c conditional_logic JSON to
rows. Strict dispatch: unknown operators / unknown top-level keys /
malformed groups FAIL the migration — Phase A seed-scan confirmed
the catalogue parity, so any drift is a data bug to fix at source,
not silently absorb. Rollback rebuilds canonical JSON and clears
the relational tree.

FormFieldService.create/update route `conditional_logic` through
the new service (matching the extract-and-delegate pattern from
WS-5a bindings and WS-5b validation rules). Snapshot writer + both
resources (FormFieldResource, PublicFormSchemaResource) read via
`toJsonShape(rootConditionalLogicGroup())` — byte-for-byte parity
with the pre-WS-5c JSON contract.

InvalidConditionalLogicSpecException handled in FormFieldController
as 422, same as FrozenSchemaException / CyclicDependencyException.

Tests: 20 new under tests/Feature/FormBuilder/ConditionalLogic/
(service, cycle detection, backfill forward+rollback+failure cases,
snapshot + resource parity). FormFieldApiTest cyclic rejection test
rewritten to use the new factory state. Rollback step counts in
WS-5a/b migration tests bumped +1 for the new backfill migration.
Baseline 1122 → 1142 green (3032 → 3085 assertions).

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-24 23:56:39 +02:00

125 lines
4.7 KiB
PHP

<?php
declare(strict_types=1);
namespace Tests\Feature\FormBuilder\ConditionalLogic;
use App\Exceptions\FormBuilder\CyclicDependencyException;
use App\Models\FormBuilder\FormField;
use App\Models\FormBuilder\FormSchema;
use App\Models\Organisation;
use App\Services\FormBuilder\FormFieldConditionalLogicService;
use Illuminate\Foundation\Testing\RefreshDatabase;
use Tests\TestCase;
/**
* Cycle detection (ARCH §8). Migrated from
* FormFieldService::assertNoConditionalCycle (pre-WS-5c) to
* FormFieldConditionalLogicService::assertNoCycles. Contract preserved;
* implementation now reads the relational tree instead of walking JSON.
*/
final class ConditionalLogicCycleDetectionTest extends TestCase
{
use RefreshDatabase;
public function test_two_node_cycle_rejected(): void
{
[$schema, $fieldA, $fieldB] = $this->seedSchemaWithTwoFields();
$service = app(FormFieldConditionalLogicService::class);
// A depends on B.
$service->replaceLogic($fieldA, [
'operator' => 'all',
'children' => [
['field_slug' => $fieldB->slug, 'operator' => 'equals', 'value' => 'y'],
],
]);
// Proposing B depends on A would close the A → B → A cycle.
$this->expectException(CyclicDependencyException::class);
$service->assertNoCycles($fieldB, [
'operator' => 'all',
'children' => [
['field_slug' => $fieldA->slug, 'operator' => 'equals', 'value' => 'x'],
],
]);
}
public function test_three_node_cycle_rejected(): void
{
[$schema, $fieldA, $fieldB, $fieldC] = $this->seedSchemaWithThreeFields();
$service = app(FormFieldConditionalLogicService::class);
// A → B, B → C
$service->replaceLogic($fieldA, [
'operator' => 'all',
'children' => [['field_slug' => $fieldB->slug, 'operator' => 'equals', 'value' => 'y']],
]);
$service->replaceLogic($fieldB, [
'operator' => 'all',
'children' => [['field_slug' => $fieldC->slug, 'operator' => 'equals', 'value' => 'y']],
]);
// Proposing C → A closes A → B → C → A.
$this->expectException(CyclicDependencyException::class);
$service->assertNoCycles($fieldC, [
'operator' => 'all',
'children' => [['field_slug' => $fieldA->slug, 'operator' => 'equals', 'value' => 'x']],
]);
}
public function test_diamond_is_accepted_no_cycle(): void
{
[$schema, $fieldA, $fieldB, $fieldC] = $this->seedSchemaWithThreeFields();
$service = app(FormFieldConditionalLogicService::class);
// A → B and C → B (two fields depend on the same field, no cycle).
$service->replaceLogic($fieldA, [
'operator' => 'all',
'children' => [['field_slug' => $fieldB->slug, 'operator' => 'equals', 'value' => 'y']],
]);
$service->assertNoCycles($fieldC, [
'operator' => 'all',
'children' => [['field_slug' => $fieldB->slug, 'operator' => 'equals', 'value' => 'y']],
]);
$this->expectNotToPerformAssertions();
}
public function test_self_reference_rejected(): void
{
[$schema, $fieldA] = $this->seedSchemaWithTwoFields();
$service = app(FormFieldConditionalLogicService::class);
$this->expectException(CyclicDependencyException::class);
$service->assertNoCycles($fieldA, [
'operator' => 'all',
'children' => [['field_slug' => $fieldA->slug, 'operator' => 'equals', 'value' => 'x']],
]);
}
/** @return array{0:FormSchema,1:FormField,2:FormField} */
private function seedSchemaWithTwoFields(): array
{
$org = Organisation::factory()->create();
$schema = FormSchema::factory()->create(['organisation_id' => $org->id]);
$fieldA = FormField::factory()->create(['form_schema_id' => $schema->id, 'slug' => 'a']);
$fieldB = FormField::factory()->create(['form_schema_id' => $schema->id, 'slug' => 'b']);
return [$schema, $fieldA, $fieldB];
}
/** @return array{0:FormSchema,1:FormField,2:FormField,3:FormField} */
private function seedSchemaWithThreeFields(): array
{
$org = Organisation::factory()->create();
$schema = FormSchema::factory()->create(['organisation_id' => $org->id]);
$fieldA = FormField::factory()->create(['form_schema_id' => $schema->id, 'slug' => 'a']);
$fieldB = FormField::factory()->create(['form_schema_id' => $schema->id, 'slug' => 'b']);
$fieldC = FormField::factory()->create(['form_schema_id' => $schema->id, 'slug' => 'c']);
return [$schema, $fieldA, $fieldB, $fieldC];
}
}