nur-sery/src/schema/_scalar/ScalarSchema.php

200 lines
6.8 KiB
PHP
Raw Normal View History

2023-11-24 16:50:05 +04:00
<?php
2023-11-28 00:20:42 +04:00
namespace nur\sery\schema\_scalar;
2023-11-24 16:50:05 +04:00
2023-11-24 22:36:33 +04:00
use nulib\cl;
use nur\sery\schema\ref\ref_schema;
use nur\sery\schema\ref\ref_types;
2023-11-27 22:39:35 +04:00
use nur\sery\schema\Schema;
use nur\sery\schema\SchemaException;
2023-11-27 18:57:07 +04:00
use nur\sery\schema\types\tarray;
use nur\sery\schema\types\tbool;
use nur\sery\schema\types\tcallable;
use nur\sery\schema\types\tcontent;
use nur\sery\schema\types\tpkey;
use nur\sery\schema\types\tstring;
2023-11-27 22:39:35 +04:00
use nur\sery\schema\Value;
2023-11-24 22:36:33 +04:00
/**
* Class ScalarSchema
*
* @property-read array $type
* @property-read mixed $default
* @property-read string|null $title
* @property-read bool $required
* @property-read bool $nullable
* @property-read string|array|null $desc
* @property-read callable|null $analyzerFunc
* @property-read callable|null $extractorFunc
* @property-read callable|null $parserFunc
* @property-read callable|null $normalizerFunc
* @property-read array|null $messages
* @property-read callable|null $formatterFunc
* @property-read mixed $format
* @property-read array $nature
* @property-read string|int|null $name
2023-11-27 18:57:07 +04:00
* @property-read string|array|null $pkey
* @property-read string|null $header
* @property-read bool|null $composite
2023-11-24 22:36:33 +04:00
*/
2023-11-24 16:50:05 +04:00
class ScalarSchema extends Schema {
/** @var array meta-schema d'un schéma de nature scalaire */
const METASCHEMA = ref_schema::SCALAR_METASCHEMA;
2023-11-24 22:36:33 +04:00
const METASCHEMA_KEYS = [
"type", "default", "title", "required", "nullable", "desc",
"analyzer_func", "extractor_func", "parser_func", "normalizer_func", "messages",
"formatter_func", "format",
2023-11-27 18:57:07 +04:00
"", "name", "pkey", "header", "composite",
2023-11-24 22:36:33 +04:00
];
2023-11-24 16:50:05 +04:00
/**
* indiquer si $definition est une définition de schéma scalaire que
* {@link normalize()} pourrait normaliser
*/
static function isa_definition($definition): bool {
# chaine ou null
if ($definition === null) return true;
if (is_string($definition)) return true;
if (!is_array($definition)) return false;
# nature explicitement spécifiée
if (array_key_exists("", $definition)) {
$nature = $definition[""];
if ($nature === "scalar") return true;
if (is_array($nature)
&& array_key_exists(0, $nature)
&& $nature[0] === "scalar") {
return true;
}
return false;
}
# un unique élément chaine à l'index 0
$count = count($definition);
$haveIndex0 = array_key_exists(0, $definition);
if ($count == 1 && $haveIndex0
&& ($definition[0] === null || is_string($definition[0]))) {
return true;
}
# un élément à l'index 0, et d'autres éléments
return $haveIndex0 && $count > 1;
}
2023-11-27 18:57:07 +04:00
static function normalize($definition, $definitionKey=null): array {
2023-11-24 22:36:33 +04:00
if (!is_array($definition)) $definition = [$definition];
# s'assurer que toutes les clés existent avec leur valeur par défaut
$index = 0;
2023-11-27 18:57:07 +04:00
foreach (self::METASCHEMA_KEYS as $pkey) {
if (!array_key_exists($pkey, $definition)) {
2023-11-24 22:36:33 +04:00
if (array_key_exists($index, $definition)) {
2023-11-27 18:57:07 +04:00
$definition[$pkey] = $definition[$index];
2023-11-24 22:36:33 +04:00
unset($definition[$index]);
$index++;
} else {
2023-11-27 18:57:07 +04:00
$definition[$pkey] = self::METASCHEMA[$pkey][1];
2023-11-24 22:36:33 +04:00
}
}
}
# réordonner les clés numériques
if (cl::have_num_keys($definition)) {
$keys = array_keys($definition);
$index = 0;
2023-11-27 18:57:07 +04:00
foreach ($keys as $pkey) {
if (!is_int($pkey)) continue;
$definition[$index] = $definition[$pkey];
unset($definition[$pkey]);
2023-11-24 22:36:33 +04:00
$index++;
}
}
# type
$types = [];
$deftype = $definition["type"];
$nullable = $definition["nullable"];
if ($deftype === null) {
$types[] = null;
$nullable = true;
} else {
if (!is_array($deftype)) {
if (!is_string($deftype)) throw SchemaException::invalid_type($deftype);
$deftype = explode("|", $deftype);
}
foreach ($deftype as $type) {
if ($type === null || $type === "null") {
$nullable = true;
continue;
}
if (!is_string($type)) throw SchemaException::invalid_type($type);
if (substr($type, 0, 1) == "?") {
$type = substr($type, 1);
$nullable = true;
}
if ($type === "") throw SchemaException::invalid_type($type);
$type = cl::get(ref_types::ALIASES, $type, $type);
$types = array_merge($types, explode("|", $type));
}
if (!$types) throw SchemaException::invalid_schema("scalar: type is required");
$types = array_keys(array_fill_keys($types, true));
}
$definition["type"] = $types;
$definition["nullable"] = $nullable;
# nature
$nature = $definition[""];
2023-11-27 18:57:07 +04:00
tarray::ensure_array($nature);
2023-11-24 22:36:33 +04:00
if (!array_key_exists(0, $nature) || $nature[0] !== "scalar") {
throw SchemaException::invalid_schema("expected scalar nature");
}
$definition[""] = $nature;
2023-11-27 18:57:07 +04:00
# name, pkey, header
2023-11-25 10:04:24 +04:00
$name = $definition["name"];
2023-11-27 18:57:07 +04:00
$pkey = $definition["pkey"];
2023-11-25 10:04:24 +04:00
$header = $definition["header"];
2023-11-27 18:57:07 +04:00
if ($name === null) $name = $definitionKey;
tstring::ensure_nstring($name);
tpkey::ensure_npkey($pkey);
tstring::ensure_nstring($header);
if ($pkey === null) $pkey = $name;
2023-11-25 10:04:24 +04:00
if ($header === null) $header = $name;
$definition["name"] = $name;
2023-11-27 18:57:07 +04:00
$definition["pkey"] = $pkey;
2023-11-25 10:04:24 +04:00
$definition["header"] = $header;
2023-11-24 22:36:33 +04:00
# autres éléments
2023-11-27 18:57:07 +04:00
tstring::ensure_nstring($definition["title"]);
tbool::ensure_bool($definition["required"]);
tbool::ensure_bool($definition["nullable"]);
tcontent::ensure_ncontent($definition["desc"]);
tcallable::ensure_ncallable($definition["analyzer_func"]);
tcallable::ensure_ncallable($definition["extractor_func"]);
tcallable::ensure_ncallable($definition["parser_func"]);
tcallable::ensure_ncallable($definition["normalizer_func"]);
tarray::ensure_narray($definition["messages"]);
tcallable::ensure_ncallable($definition["formatter_func"]);
tbool::ensure_nbool($definition["composite"]);
2023-11-24 22:36:33 +04:00
return $definition;
2023-11-24 16:50:05 +04:00
}
2023-11-28 08:20:33 +04:00
function __construct($definition=null, $definitionKey=null, bool $normalize=true) {
2023-11-24 16:50:05 +04:00
if ($definition === null) $definition = static::SCHEMA;
2023-11-28 08:20:33 +04:00
if ($normalize) $definition = self::normalize($definition, $definitionKey);
2023-11-24 16:50:05 +04:00
$this->definition = $definition;
}
2023-11-24 22:36:33 +04:00
2023-11-25 10:04:24 +04:00
function isScalar(?ScalarSchema &$scalar=null): bool {
$scalar = $this;
2023-11-24 22:36:33 +04:00
return true;
}
2023-11-25 10:04:24 +04:00
2023-11-28 00:20:42 +04:00
function newValue(?Value &$value=null, &$dest=null, $key=null): Value {
2023-11-27 22:39:35 +04:00
if (!($value instanceof ScalarValue)) $value = new ScalarValue($this);
return $value->reset($dest, $key);
2023-11-25 10:04:24 +04:00
}
2023-11-27 18:57:07 +04:00
#############################################################################
# key & properties
const _PROPERTY_PKEYS = [
"analyzerFunc" => "analyzer_func",
"extractorFunc" => "extractor_func",
"parserFunc" => "parser_func",
"normalizerFunc" => "normalizer_func",
"formatterFunc" => "formatter_func",
"nature" => ["", 0],
];
2023-11-24 16:50:05 +04:00
}