Per-purpose schema validation composes a PurposeGuardProvider returning a list of guards. Errors collected (not first-fail) so the builder UI surfaces every issue per save. ConditionalRequirement composes higher- order without proliferating one-off classes. RequiresIdentityKeyBinding checks the is_identity_key flag specifically; the binding-existence check is handled additively by the existing assertRequiredBindingsPresent in FormSchemaService. SchemaHasLinkedEvent checks owner_type='event' + owner_id (FormSchema uses polymorphic owner; there is no direct event_id column). i18n messages live in lang/nl/form_builder_publish_guards.php. Refs: RFC-WS-6.md §3 (Q13), §4 (V1, V3) Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
71 lines
3.1 KiB
PHP
71 lines
3.1 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Tests\Unit\FormBuilder\Publishing;
|
|
|
|
use App\FormBuilder\Publishing\IdentityKeyBindingsOnlyInFirstSection;
|
|
use App\Models\FormBuilder\FormField;
|
|
use App\Models\FormBuilder\FormFieldBinding;
|
|
use App\Models\FormBuilder\FormSchema;
|
|
use App\Models\FormBuilder\FormSchemaSection;
|
|
use Illuminate\Foundation\Testing\RefreshDatabase;
|
|
use Tests\TestCase;
|
|
|
|
final class IdentityKeyBindingsOnlyInFirstSectionTest extends TestCase
|
|
{
|
|
use RefreshDatabase;
|
|
|
|
public function test_no_op_when_section_level_submit_is_false(): void
|
|
{
|
|
$schema = FormSchema::factory()->create(['section_level_submit' => false]);
|
|
FormSchemaSection::factory()->create(['form_schema_id' => $schema->id, 'sort_order' => 0]);
|
|
$section2 = FormSchemaSection::factory()->create(['form_schema_id' => $schema->id, 'sort_order' => 1]);
|
|
$field = FormField::factory()->create([
|
|
'form_schema_id' => $schema->id,
|
|
'form_schema_section_id' => $section2->id, // identity key in 2nd section
|
|
]);
|
|
FormFieldBinding::factory()->forField($field)->entityOwned('person', 'email')
|
|
->create(['is_identity_key' => true]);
|
|
$schema->load(['fields.bindings', 'sections']);
|
|
|
|
$result = (new IdentityKeyBindingsOnlyInFirstSection())->evaluate($schema);
|
|
$this->assertTrue($result->passed);
|
|
}
|
|
|
|
public function test_passes_when_identity_key_in_first_section(): void
|
|
{
|
|
$schema = FormSchema::factory()->create(['section_level_submit' => true]);
|
|
$section1 = FormSchemaSection::factory()->create(['form_schema_id' => $schema->id, 'sort_order' => 0]);
|
|
FormSchemaSection::factory()->create(['form_schema_id' => $schema->id, 'sort_order' => 1]);
|
|
$field = FormField::factory()->create([
|
|
'form_schema_id' => $schema->id,
|
|
'form_schema_section_id' => $section1->id,
|
|
]);
|
|
FormFieldBinding::factory()->forField($field)->entityOwned('artist', 'email')
|
|
->create(['is_identity_key' => true]);
|
|
$schema->load(['fields.bindings', 'sections']);
|
|
|
|
$result = (new IdentityKeyBindingsOnlyInFirstSection())->evaluate($schema);
|
|
$this->assertTrue($result->passed);
|
|
}
|
|
|
|
public function test_fails_when_identity_key_in_later_section(): void
|
|
{
|
|
$schema = FormSchema::factory()->create(['section_level_submit' => true]);
|
|
FormSchemaSection::factory()->create(['form_schema_id' => $schema->id, 'sort_order' => 0]);
|
|
$section2 = FormSchemaSection::factory()->create(['form_schema_id' => $schema->id, 'sort_order' => 1]);
|
|
$field = FormField::factory()->create([
|
|
'form_schema_id' => $schema->id,
|
|
'form_schema_section_id' => $section2->id,
|
|
]);
|
|
FormFieldBinding::factory()->forField($field)->entityOwned('artist', 'email')
|
|
->create(['is_identity_key' => true]);
|
|
$schema->load(['fields.bindings', 'sections']);
|
|
|
|
$result = (new IdentityKeyBindingsOnlyInFirstSection())->evaluate($schema);
|
|
$this->assertFalse($result->passed);
|
|
$this->assertSame((string) $field->id, $result->offendingFormFieldId);
|
|
}
|
|
}
|