JsonSchemaTestTrait.php

Namespace

Drupal\Tests\serialization\Traits

File

core/modules/serialization/tests/src/Traits/JsonSchemaTestTrait.php

View source
<?php

declare (strict_types=1);
namespace Drupal\Tests\serialization\Traits;

use Drupal\serialization\Normalizer\PrimitiveDataNormalizer;
use JsonSchema\Validator;
use Prophecy\Prophecy\ObjectProphecy;
use Symfony\Component\Serializer\Normalizer\NormalizerInterface;

/**
 * Trait for testing JSON Schema validity and fit to sample data.
 *
 * In most cases, tests need only implement the abstract method for providing
 * a full set of representative normalized values.
 */
trait JsonSchemaTestTrait {
    
    /**
     * Format that should be used when performing test normalizations.
     */
    protected function getJsonSchemaTestNormalizationFormat() : ?string {
        return NULL;
    }
    
    /**
     * Data provider for ::testNormalizedValuesAgainstJsonSchema.
     *
     * @return array
     *   Array of possible normalized values to validate the JSON schema against.
     */
    public static abstract function jsonSchemaDataProvider() : array;
    
    /**
     * Method to make prophecy public for use in data provider closures.
     *
     * @return \Prophecy\Prophecy\ObjectProphecy<object>
     */
    public function doProphesize(?string $classOrInterface = NULL) : ObjectProphecy {
        return $this->prophesize($classOrInterface);
    }
    
    /**
     * Test that a valid schema is returned for the explicitly supported types.
     *
     * This is in many cases an interface, which would not be normalized directly,
     * however the schema should never return an invalid type. An empty array or
     * a type with only a '$comment' member is valid.
     *
     * @dataProvider supportedTypesDataProvider
     */
    public function testSupportedTypesSchemaIsValid(string $type) : void {
        $this->doTestJsonSchemaIsValid($type, TRUE);
    }
    
    /**
     * Check a schema is valid against the meta-schema.
     *
     * @param array $defined_schema
     *   Defined schema.
     * @param bool $accept_no_schema_type
     *   Whether to accept a schema with no meaningful type construct.
     */
    protected function doCheckSchemaAgainstMetaSchema(array $defined_schema, bool $accept_no_schema_type = FALSE) : void {
        $validator = $this->getValidator();
        // Ensure the schema contains a meaningful type construct.
        if (!$accept_no_schema_type) {
            $this->assertFalse(empty(array_filter(array_keys($defined_schema), fn($key) => in_array($key, [
                'type',
                'allOf',
                'oneOf',
                'anyOf',
                'not',
                '$ref',
            ]))));
        }
        // All associative arrays must be encoded as objects.
        $schema = json_decode(json_encode($defined_schema));
        $validator->validate($schema, (object) [
            '$ref' => 'file://' . __DIR__ . '/../../../src/json-schema-draft-04-meta-schema.json',
        ]);
        $this->assertTrue($validator->isValid());
    }
    
    /**
     * Validate the normalizer's JSON schema.
     *
     * @param mixed $type
     *   Object/type being normalized.
     * @param bool $accept_no_schema_type
     *   Whether to accept a schema with no meaningful type.
     *
     * @return array
     *   Schema, so later tests can avoid retrieving it again.
     */
    public function doTestJsonSchemaIsValid(mixed $type, bool $accept_no_schema_type = FALSE) : array {
        $defined_schema = $this->getNormalizer()
            ->normalize($type, 'json_schema');
        $this->doCheckSchemaAgainstMetaSchema($defined_schema, $accept_no_schema_type);
        return $defined_schema;
    }
    
    /**
     * @return array
     *   Supported types for which to test schema generation.
     */
    public static function supportedTypesDataProvider() : array {
        return array_map(fn($type) => [
            $type,
        ], array_keys((new PrimitiveDataNormalizer())->getSupportedTypes(NULL)));
    }
    
    /**
     * Test normalized values against the JSON schema.
     *
     * @dataProvider jsonSchemaDataProvider
     */
    public function testNormalizedValuesAgainstJsonSchema(mixed $value) : void {
        // Explicitly test the JSON Schema's validity here, because it will depend
        // on the type of the data being normalized, e.g. a class implementing the
        // interface defined in ::getSupportedTypes().
        if ($value instanceof \Closure) {
            $value = $value($this);
        }
        $schema = $this->doTestJsonSchemaIsValid($value);
        $validator = $this->getValidator();
        // Test the value validates to the schema.
        // All associative arrays must be encoded as objects.
        $normalized = json_decode(json_encode($this->getNormalizationForValue($value)));
        $validator->validate($normalized, $schema);
        $this->assertSame([], $validator->getErrors(), 'Validation errors on object ' . print_r($normalized, TRUE) . ' with schema ' . print_r($schema, TRUE));
    }
    
    /**
     * Helper method to retrieve the normalizer.
     *
     * Override this method if the normalizer has a custom getter or is not
     * already present at $this->normalizer.
     *
     * @return \Symfony\Component\Serializer\Normalizer\NormalizerInterface
     *   The normalizer under test.
     */
    protected function getNormalizer() : NormalizerInterface {
        return $this->normalizer;
    }
    
    /**
     * Get the normalization for a value.
     *
     * Override this method if the normalization needs further processing, e.g.
     * in the case of JSON:API module's CacheableDependencyInterface.
     *
     * @param mixed $value
     *   Value to be normalized.
     *
     * @return mixed
     *   Final normalized value.
     */
    protected function getNormalizationForValue(mixed $value) : mixed {
        return $this->getNormalizer()
            ->normalize($value, $this->getJsonSchemaTestNormalizationFormat());
    }
    
    /**
     * Get the JSON Schema Validator.
     *
     * Override this method to add additional schema translations to the loader.
     *
     * @return \JsonSchema\Validator
     *   Schema validator.
     */
    protected function getValidator() : Validator {
        return new Validator();
    }

}

Traits

Title Deprecated Summary
JsonSchemaTestTrait Trait for testing JSON Schema validity and fit to sample data.

Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.