seed(RoleSeeder::class); $this->org = Organisation::factory()->create(); $this->actor = User::factory()->create(); $this->org->users()->attach($this->actor, ['role' => 'org_admin']); $this->actor->assignRole('org_admin'); setPermissionsTeamId($this->org->id); $this->service = $this->app->make(FormSchemaService::class); } public function test_publish_succeeds_when_all_required_bindings_are_in_relational_table(): void { $schema = $this->service->create( $this->org, ['name' => 'ER', 'purpose' => FormPurpose::EVENT_REGISTRATION->value], $this->actor, ); // RFC v1.1 §3 Q8 addendum: event_registration schemas need a // default_crowd_type_id (RequiresDefaultCrowdType publish guard). $crowdType = \App\Models\CrowdType::factory()->create([ 'organisation_id' => $this->org->id, ]); $schema->default_crowd_type_id = $crowdType->id; $schema->save(); // WS-6 publish guards require: EMAIL field type, identity_key flag // on person.email, unique trust levels per (entity, attribute). $emailField = FormField::factory()->create([ 'form_schema_id' => $schema->id, 'field_type' => FormFieldType::EMAIL->value, ]); FormFieldBinding::factory()->forField($emailField)->entityOwned('person', 'email') ->create(['is_identity_key' => true, 'trust_level' => 80]); $firstField = FormField::factory()->create(['form_schema_id' => $schema->id]); FormFieldBinding::factory()->forField($firstField)->entityOwned('person', 'first_name') ->create(['trust_level' => 70]); $lastField = FormField::factory()->create(['form_schema_id' => $schema->id]); FormFieldBinding::factory()->forField($lastField)->entityOwned('person', 'last_name') ->create(['trust_level' => 60]); $published = $this->service->publish($schema->fresh(), $this->actor); $this->assertTrue((bool) $published->is_published); } public function test_publish_fails_when_required_binding_missing_reports_exact_paths(): void { $schema = $this->service->create( $this->org, ['name' => 'ER-partial', 'purpose' => FormPurpose::EVENT_REGISTRATION->value], $this->actor, ); FormField::factory()->withEntityBinding('person', 'email')->create(['form_schema_id' => $schema->id]); try { $this->service->publish($schema->fresh(), $this->actor); $this->fail('Expected PurposeRequirementsNotMetException'); } catch (PurposeRequirementsNotMetException $e) { $this->assertSame('event_registration', $e->purposeSlug); $this->assertSame(['person.first_name', 'person.last_name'], $e->missingBindings); } } public function test_publish_ignores_bindings_belonging_to_other_schemas(): void { $schemaA = $this->service->create( $this->org, ['name' => 'A', 'purpose' => FormPurpose::SUPPLIER_INTAKE->value], $this->actor, ); $schemaB = $this->service->create( $this->org, ['name' => 'B', 'purpose' => FormPurpose::SUPPLIER_INTAKE->value], $this->actor, ); FormField::factory()->withEntityBinding('company', 'contact_email')->create(['form_schema_id' => $schemaB->id]); try { $this->service->publish($schemaA->fresh(), $this->actor); $this->fail('Expected PurposeRequirementsNotMetException'); } catch (PurposeRequirementsNotMetException $e) { $this->assertSame('supplier_intake', $e->purposeSlug); $this->assertSame(['company.name'], $e->missingBindings); } } }