This adds support for JSONSchema style property declarations with nested schemas. This is a step towards using more nested structured for configuration, rather than adding to the over 700 keys already defined in the main config schema. Defaults from property schemas are aggregated into a default value in the top level schema. Descriptions are however not yet aggregated. Change-Id: Iaf46a9ecc83bee3566098c56137a1be66bff2ab9
160 lines
3.9 KiB
PHP
160 lines
3.9 KiB
PHP
<?php
|
|
|
|
namespace MediaWiki\Settings\Source;
|
|
|
|
use InvalidArgumentException;
|
|
|
|
/**
|
|
* Trait for dealing with JSON Schema structures and types.
|
|
*
|
|
* @since 1.39
|
|
*/
|
|
trait JsonSchemaTrait {
|
|
|
|
/**
|
|
* Converts a JSON Schema type to a PHPDoc type.
|
|
*
|
|
* @param string|string[] $jsonSchemaType A JSON Schema type
|
|
*
|
|
* @return string A PHPDoc type
|
|
*/
|
|
private static function jsonToPhpDoc( $jsonSchemaType ) {
|
|
static $phpTypes = [
|
|
'array' => 'array',
|
|
'object' => 'array', // could be optional
|
|
'number' => 'float',
|
|
'double' => 'float', // for good measure
|
|
'boolean' => 'bool',
|
|
'integer' => 'int',
|
|
];
|
|
|
|
if ( $jsonSchemaType === null ) {
|
|
throw new InvalidArgumentException( 'The type name cannot be null! Use "null" instead.' );
|
|
}
|
|
|
|
$nullable = false;
|
|
if ( is_array( $jsonSchemaType ) ) {
|
|
$nullIndex = array_search( 'null', $jsonSchemaType );
|
|
if ( $nullIndex !== false ) {
|
|
$nullable = true;
|
|
unset( $jsonSchemaType[$nullIndex] );
|
|
}
|
|
|
|
$jsonSchemaType = array_map( [ self::class, 'jsonToPhpDoc' ], $jsonSchemaType );
|
|
$type = implode( '|', $jsonSchemaType );
|
|
} else {
|
|
$type = $phpTypes[ strtolower( $jsonSchemaType ) ] ?? $jsonSchemaType;
|
|
}
|
|
|
|
if ( $nullable ) {
|
|
$type = "?$type";
|
|
}
|
|
|
|
return $type;
|
|
}
|
|
|
|
/**
|
|
* @param string|string[] $phpDocType The PHPDoc type
|
|
*
|
|
* @return string|string[] A JSON Schema type
|
|
*/
|
|
private static function phpDocToJson( $phpDocType ) {
|
|
static $jsonTypes = [
|
|
'list' => 'array',
|
|
'dict' => 'object',
|
|
'map' => 'object',
|
|
'stdclass' => 'object',
|
|
'int' => 'integer',
|
|
'float' => 'number',
|
|
'bool' => 'boolean',
|
|
'false' => 'boolean',
|
|
];
|
|
|
|
if ( $phpDocType === null ) {
|
|
throw new InvalidArgumentException( 'The type name cannot be null! Use "null" instead.' );
|
|
}
|
|
|
|
if ( is_array( $phpDocType ) ) {
|
|
$types = $phpDocType;
|
|
} else {
|
|
$types = explode( '|', trim( $phpDocType ) );
|
|
}
|
|
|
|
$nullable = false;
|
|
foreach ( $types as $i => $t ) {
|
|
if ( str_starts_with( $t, '?' ) ) {
|
|
$nullable = true;
|
|
$t = substr( $t, 1 );
|
|
}
|
|
|
|
$types[$i] = $jsonTypes[ strtolower( $t ) ] ?? $t;
|
|
}
|
|
|
|
if ( $nullable ) {
|
|
$types[] = 'null';
|
|
}
|
|
|
|
$types = array_unique( $types );
|
|
|
|
if ( count( $types ) === 1 ) {
|
|
return reset( $types );
|
|
}
|
|
|
|
return $types;
|
|
}
|
|
|
|
/**
|
|
* Recursively applies phpDocToJson() to type declarations in a JSON schema.
|
|
*
|
|
* @param array $schema JSON Schema structure with PHPDoc types
|
|
*
|
|
* @return array JSON Schema structure using only proper JSON types
|
|
*/
|
|
private static function normalizeJsonSchema( array $schema ): array {
|
|
if ( isset( $schema['type'] ) ) {
|
|
// Support PHP Doc style types, for convenience.
|
|
$schema['type'] = self::phpDocToJson( $schema['type'] );
|
|
}
|
|
|
|
if ( isset( $schema['additionalProperties'] ) && is_array( $schema['additionalProperties'] ) ) {
|
|
$schema['additionalProperties'] =
|
|
self::normalizeJsonSchema( $schema['additionalProperties'] );
|
|
}
|
|
|
|
if ( isset( $schema['items'] ) && is_array( $schema['items'] ) ) {
|
|
$schema['items'] = self::normalizeJsonSchema( $schema['items'] );
|
|
}
|
|
|
|
if ( isset( $schema['properties'] ) && is_array( $schema['properties'] ) ) {
|
|
foreach ( $schema['properties'] as $name => $propSchema ) {
|
|
$schema['properties'][$name] = self::normalizeJsonSchema( $propSchema );
|
|
}
|
|
}
|
|
|
|
return $schema;
|
|
}
|
|
|
|
/**
|
|
* Returns the default value from the given schema structure.
|
|
* If the schema defines properties, the default value of each
|
|
* property is determined recursively, and the collected into a
|
|
* the top level default, which in that case will be a map
|
|
* (that is, a JSON object).
|
|
*
|
|
* @param array $schema
|
|
* @return mixed The default specified by $schema, or null if no default
|
|
* is defined.
|
|
*/
|
|
private static function getDefaultFromJsonSchema( array $schema ) {
|
|
$default = $schema['default'] ?? null;
|
|
|
|
foreach ( $schema['properties'] ?? [] as $name => $sch ) {
|
|
$def = self::getDefaultFromJsonSchema( $sch );
|
|
|
|
$default[$name] = $def;
|
|
}
|
|
|
|
return $default;
|
|
}
|
|
|
|
}
|