set($class, $field, $value); } $data = RelationValidationService::singleton()->inspectClasses([ Team::class, Member::class, Hat::class, Freelancer::class, ]); $this->assertSame($expected, $data); } /** * @param string $class * @param string|null $relation * @param array $config * @param bool $expected */ #[DataProvider('ignoredClassesProvider')] public function testIgnoredClass(string $class, ?string $relation, array $config, bool $expected): void { if (!class_exists($class)) { $this->markTestSkipped("This test requires the $class class"); } $service = RelationValidationService::singleton(); foreach ($config as $name => $value) { $service->config()->set($name, $value); } $result = $service->isIgnored($class, $relation); $this->assertEquals($expected, $result); } public static function validateCasesProvider(): array { return [ 'correct setup' => [ null, null, [], [], ], 'ambiguous has_one - no relation name' => [ Hat::class, 'belongs_to', [ 'Hatter' => Member::class, ], [ 'SilverStripe\Dev\Tests\Validation\Member / Hat : Back relation not found or ambiguous (needs class.relation format)', 'SilverStripe\Dev\Tests\Validation\Hat / Hatter : Relation is not in the expected format (needs class.relation format)', ], ], 'ambiguous has_one - incorrect relation name' => [ Hat::class, 'belongs_to', [ 'Hatter' => Member::class . '.ObviouslyWrong', ], [ 'SilverStripe\Dev\Tests\Validation\Member / Hat : Back relation not found or ambiguous (needs class.relation format)', 'SilverStripe\Dev\Tests\Validation\Hat / Hatter : Back relation not found', ], ], 'ambiguous has_one - too many relations' => [ Hat::class, 'belongs_to', [ 'Hatter' => Member::class . '.Hat', 'HatterCopy' => Member::class . '.Hat', ], [ 'SilverStripe\Dev\Tests\Validation\Member / Hat : Back relation is ambiguous', ], ], 'polymorphic has one' => [ Team::class, 'has_one', [ 'SingleMember' => DataObject::class, ], [], ], 'invalid has one' => [ Member::class, 'has_one', [ 'HomeTeam' => Team::class . '.UnnecessaryRelation', 'Hat' => Hat::class, ], [ 'SilverStripe\Dev\Tests\Validation\Member / HomeTeam : Relation SilverStripe\Dev\Tests\Validation\Team.UnnecessaryRelation is not in the expected format (needs class only format).' ], ], 'has_one missing class in array config' => [ Team::class, 'has_one', [ 'SingleMember' => [ DataObjectSchema::HAS_ONE_MULTI_RELATIONAL => true, ], ], [ 'SilverStripe\Dev\Tests\Validation\Team / SingleMember : No class has been defined for this relation.' ], ], 'multi-relational has_one should be polymorphic' => [ Team::class, 'has_one', [ 'SingleMember' => [ 'class' => Member::class, DataObjectSchema::HAS_ONE_MULTI_RELATIONAL => true, ], ], [ 'SilverStripe\Dev\Tests\Validation\Team / SingleMember : has_one relation that can handle multiple reciprocal has_many relations must be polymorphic.' ], ], 'has_one defines class in array config' => [ Team::class, 'has_one', [ 'SingleMember' => [ 'class' => Member::class, ], ], // Note there's no message about the has_one class, which is technically correctly defined. // The bad thing now is just that we still have multiple has_many relations pointing at it. [ 'SilverStripe\Dev\Tests\Validation\Team / SingleMember : Back relation is ambiguous' ], ], 'ambiguous has_many - no relation name' => [ Team::class, 'has_many', [ 'Members' => Member::class, 'FreelancerMembers' => Freelancer::class . '.TemporaryTeam', ], [ 'SilverStripe\Dev\Tests\Validation\Team / Members : Relation is not in the expected format (needs class.relation format)', 'SilverStripe\Dev\Tests\Validation\Member / HomeTeam : Back relation not found or ambiguous (needs class.relation format)', ], ], 'ambiguous has_many - incorrect relation name' => [ Team::class, 'has_many', [ 'Members' => Member::class . '.ObviouslyWrong', 'FreelancerMembers' => Freelancer::class . '.TemporaryTeam', ], [ 'SilverStripe\Dev\Tests\Validation\Team / Members : Back relation not found or ambiguous (needs class.relation format)', 'SilverStripe\Dev\Tests\Validation\Member / HomeTeam : Back relation not found or ambiguous (needs class.relation format)', ], ], 'ambiguous has_many - too many relations' => [ Team::class, 'has_many', [ 'Members' => Member::class . '.HomeTeam', 'MembersCopy' => Member::class . '.HomeTeam', 'FreelancerMembers' => Freelancer::class . '.TemporaryTeam', ], [ 'SilverStripe\Dev\Tests\Validation\Member / HomeTeam : Back relation is ambiguous', ], ], 'ambiguous many_many - no relation name' => [ Hat::class, 'belongs_many_many', [ 'TeamHats' => Team::class, ], [ 'SilverStripe\Dev\Tests\Validation\Team / ReserveHats : Back relation not found or ambiguous (needs class.relation format)', 'SilverStripe\Dev\Tests\Validation\Hat / TeamHats : Relation is not in the expected format (needs class.relation format)', ], ], 'ambiguous many_many - incorrect relation name' => [ Hat::class, 'belongs_many_many', [ 'TeamHats' => Team::class . '.ObviouslyWrong', ], [ 'SilverStripe\Dev\Tests\Validation\Team / ReserveHats : Back relation not found or ambiguous (needs class.relation format)', 'SilverStripe\Dev\Tests\Validation\Hat / TeamHats : Back relation not found', ], ], 'ambiguous many_many - too many relations' => [ Hat::class, 'belongs_many_many', [ 'TeamHats' => Team::class . '.ReserveHats', 'TeamHatsCopy' => Team::class . '.ReserveHats', ], [ 'SilverStripe\Dev\Tests\Validation\Team / ReserveHats : Back relation is ambiguous', ], ], 'ambiguous many_many through - no relation name' => [ Member::class, 'belongs_many_many', [ 'FreelancerTeams' => Team::class, ], [ 'SilverStripe\Dev\Tests\Validation\Team / Freelancers : Back relation not found or ambiguous (needs class.relation format)', 'SilverStripe\Dev\Tests\Validation\Member / FreelancerTeams : Relation is not in the expected format (needs class.relation format)', ], ], 'ambiguous many_many through - incorrect relation name' => [ Member::class, 'belongs_many_many', [ 'FreelancerTeams' => Team::class . '.ObviouslyWrong', ], [ 'SilverStripe\Dev\Tests\Validation\Team / Freelancers : Back relation not found or ambiguous (needs class.relation format)', 'SilverStripe\Dev\Tests\Validation\Member / FreelancerTeams : Back relation not found', ], ], 'ambiguous many_many through - too many relations' => [ Member::class, 'belongs_many_many', [ 'FreelancerTeams' => Team::class . '.Freelancers', 'FreelancerTeamsCopy' => Team::class . '.Freelancers', ], [ 'SilverStripe\Dev\Tests\Validation\Team / Freelancers : Back relation is ambiguous', ], ], ]; } public static function ignoredClassesProvider(): array { return [ 'class default' => [ Team::class, null, [], true, ], 'class relation default' => [ Team::class, 'Members', [], true, ], 'page should by included by default (empty namespace)' => [ Page::class, null, [], false, ], 'class relation via allow rules' => [ Team::class, 'Members', [ 'allow_rules' => ['app' => 'SilverStripe\Dev\Tests\Validation'], ], false, ], 'class included via allow rules but overwritten by deny rules' => [ Team::class, null, [ 'allow_rules' => ['app' => 'SilverStripe\Dev\Tests\Validation'], 'deny_rules' => [Team::class], ], true, ], 'class relation included via allow rules but overwritten by deny rules' => [ Team::class, 'Members', [ 'allow_rules' => ['app' => 'SilverStripe\Dev\Tests\Validation'], 'deny_rules' => [Team::class], ], true, ], 'class relation included via allow rules but overwritten by deny relations' => [ Team::class, 'Members', [ 'allow_rules' => ['app' => 'SilverStripe\Dev\Tests\Validation'], 'deny_relations' => [Team::class . '.Members'], ], true, ], 'class relation included via allow rules and not overwritten by deny relations of other class' => [ Member::class, 'HomeTeam', [ 'allow_rules' => ['app' => 'SilverStripe\Dev\Tests\Validation'], 'deny_rules' => [Team::class], 'deny_relations' => [Team::class . '.Members'], ], false, ], ]; } }