2022-02-04 14:41:09 +13:00
< ? php
namespace SilverStripe\Dev\Tests\Validation ;
use Page ;
use SilverStripe\Core\Config\Config ;
use SilverStripe\Dev\SapphireTest ;
use SilverStripe\Dev\Validation\RelationValidationService ;
2023-12-12 10:18:25 +13:00
use SilverStripe\ORM\DataObject ;
use SilverStripe\ORM\DataObjectSchema ;
2024-09-18 13:53:44 +12:00
use PHPUnit\Framework\Attributes\DataProvider ;
2022-02-04 14:41:09 +13:00
class RelationValidationTest extends SapphireTest
{
/**
* @ var array
*/
protected static $extra_dataobjects = [
Team :: class ,
Member :: class ,
Hat :: class ,
Freelancer :: class ,
];
/**
* @ param string | null $class
* @ param string | null $field
* @ param array $value
* @ param array $expected
*/
2024-09-18 13:53:44 +12:00
#[DataProvider('validateCasesProvider')]
2022-02-04 14:41:09 +13:00
public function testValidation ( ? string $class , ? string $field , array $value , array $expected ) : void
{
if ( $class && $field ) {
Config :: modify () -> 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
*/
2024-09-18 13:53:44 +12:00
#[DataProvider('ignoredClassesProvider')]
2022-02-04 14:41:09 +13:00
public function testIgnoredClass ( string $class , ? string $relation , array $config , bool $expected ) : void
{
2022-07-25 16:35:28 +12:00
if ( ! class_exists ( $class )) {
$this -> markTestSkipped ( " This test requires the $class class " );
}
2022-02-04 14:41:09 +13:00
$service = RelationValidationService :: singleton ();
foreach ( $config as $name => $value ) {
$service -> config () -> set ( $name , $value );
}
$result = $service -> isIgnored ( $class , $relation );
$this -> assertEquals ( $expected , $result );
}
2024-09-18 13:53:44 +12:00
public static function validateCasesProvider () : array
2022-02-04 14:41:09 +13:00
{
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' ,
],
],
2023-12-12 10:18:25 +13:00
'polymorphic has one' => [
Team :: class ,
'has_one' ,
[
'SingleMember' => DataObject :: class ,
],
[],
],
2022-02-04 14:41:09 +13:00
'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).'
],
],
2023-12-12 10:18:25 +13:00
'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'
],
],
2022-02-04 14:41:09 +13:00
'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' ,
],
],
];
}
2024-09-18 13:53:44 +12:00
public static function ignoredClassesProvider () : array
2022-02-04 14:41:09 +13:00
{
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 ,
],
];
}
}