1554 lines
52 KiB
PHP
1554 lines
52 KiB
PHP
|
<?php
|
||
|
namespace nur\cli;
|
||
|
|
||
|
use nur\A;
|
||
|
use nur\akey;
|
||
|
use nur\b\IllegalAccessException;
|
||
|
use nur\func;
|
||
|
use nur\md;
|
||
|
use nur\oprop;
|
||
|
use nur\ref\ref_args;
|
||
|
use nur\str;
|
||
|
use nur\types;
|
||
|
use nur\valx;
|
||
|
use stdClass;
|
||
|
|
||
|
/**
|
||
|
* Class ArgsParser: analyse d'arguments de ligne de commande
|
||
|
*
|
||
|
* == définition des arguments ==
|
||
|
*
|
||
|
* Les arguments valides sont décrits avec un tableau de définitions, qui peut
|
||
|
* contenir à la fois des clés associatives et des clés séquentielles.
|
||
|
*
|
||
|
* Les clés associatives doivent être conformes au schéma ARGS_SCHEMA et servent
|
||
|
* à donner des méta-informations sur les options et arguments valides.
|
||
|
*
|
||
|
* Les clés séquentielles sont chacune la description d'une option ou d'une
|
||
|
* commande.
|
||
|
*
|
||
|
* il est possible de grouper des options ensemble en définissant des sections.
|
||
|
* chaque section doit être conforme au schéma SECTION_SCHEMA.
|
||
|
*
|
||
|
* == définitions des options et commandes ==
|
||
|
*
|
||
|
* Chaque option ou commande est décrite par un tableau, qui peut contenir à la
|
||
|
* fois des clés associatives et des clés séquentielles.
|
||
|
*
|
||
|
* S'il n'y a aucune clé séquentielle dans une définition, alors la définition
|
||
|
* concerne la prise en charge des arguments restants sur la ligne de commande
|
||
|
* Sinon, les clés séquentielles définissent toutes les options possibles e.g
|
||
|
* '-s', '--long' ou 'command'
|
||
|
* Toutes les options d'une même définition sont strictement équivalentes
|
||
|
*
|
||
|
* Les clés associatives d'une définition doivent être conformes à OPTION_SCHEMA
|
||
|
* et décrivent l'option ou la commande:
|
||
|
*
|
||
|
* Si une option est sans argument, l'action par défaut est d'incrémenter la
|
||
|
* destination. Avec "inverse", on décrémente la destination (mais la valeur
|
||
|
* ne peux pas être inférieure à 0). Avec "value", on fixe la valeur.
|
||
|
*
|
||
|
* Si on ne fournit pas le nom de la propriété ou de la clé de destination,
|
||
|
* elle est dérivée à partir du nom de l'option longue la plus longue. Les
|
||
|
* caractères autre qu'alphanumériques sont remplacés par '_'. Si le nom final
|
||
|
* commence par un chiffre, la lettre 'p' est rajoutée en préfixe. Ainsi, la
|
||
|
* propriété par défaut pour l'option '--123' est 'p123'
|
||
|
*
|
||
|
* :: args
|
||
|
* chaque argument de "args" est le type l'argument attendu par l'option.
|
||
|
*
|
||
|
* Un tableau indique que les arguments sont facultatifs. La valeur null indique
|
||
|
* que le nombre d'arguments facultatifs est non borné. Le type de ces arguments
|
||
|
* facultatifs est celui de l'argument précédent.
|
||
|
*
|
||
|
* Sur la ligne de commande, pour arrêter la saisie des arguments facultatifs,
|
||
|
* il suffit d'utiliser '--'
|
||
|
*
|
||
|
* Par exemple, ["value"] indique que l'option prend un seul argument
|
||
|
* obligatoire, alors que ["value", ["path", "host"]] indique qu'il faut une
|
||
|
* valeur obligatoire, suivi éventuellement d'un chemin puis d'un hôte, à moins
|
||
|
* d'utiliser '--' pour arrêter la liste des arguments facultatifs.
|
||
|
*
|
||
|
* Si on spécifie un nombre, alors c'est le nombre d'argument obligatoires de
|
||
|
* type "value". Spécifier true revient à spécifier 1
|
||
|
*
|
||
|
* comme raccourci, ajouter un ':' à une option courte ou '=' à une option
|
||
|
* longue revient à spécifier true pour la valeur args si elle n'a pas déjà été
|
||
|
* définie, e.g
|
||
|
* - ["-o:"] est équivalent à ["-o", "args" => true]
|
||
|
* - ["-o:", "args" => 3] est équivalent à ["-o", "args" => 3]
|
||
|
*
|
||
|
* de même, ajouter un '::' à une option courte revient à spécifier [["value"]]
|
||
|
* pour la valeur args si elle n'a pas déjà été définie, e.g
|
||
|
* - ["-o::"] est équivalent à ["-o", "args" => [["value"]]]
|
||
|
* - ["-o::", "args" => 3] est équivalent à ["-o", "args" => 3]
|
||
|
*
|
||
|
* :: action
|
||
|
* c'est une fonction à appeler quand cette option est utilisée. Sa signature
|
||
|
* est ($dest, $value, $name, $arg, $def)
|
||
|
*
|
||
|
* Si la fonction est une méthode, elle doit être statique.
|
||
|
* Si le nom de l'action est de la forme "->method", alors la destination doit
|
||
|
* être un objet, et la méthode correspondante est appelée avec les arguments
|
||
|
* ($value, $name, $arg, $def)
|
||
|
*
|
||
|
* Certains noms d'actions sont réservés:
|
||
|
* "--inc" incrémenter la valeur (action par défaut pour une option sans argument)
|
||
|
* "--dec" décrémenter la valeur (implique "inverse" => true)
|
||
|
* "--set" forcer à la valeur spécifiée dans la clé "value"
|
||
|
* "--add" ajouter au tableau destination
|
||
|
* "--adds" ajouter chaque argument au tableau destination
|
||
|
* "--merge" fusionner dans le tableau destination
|
||
|
* "--merges" fusionner chaque argument dans le tableau destination
|
||
|
* "--meta" est utilisé en interne pour mettre à jour les champs args et command
|
||
|
*
|
||
|
* --adds (resp. --merges) sont utiles si l'option prend plusieurs arguments:
|
||
|
* l'action --add (resp. --merge) est appliqué sur chaque argument de l'option
|
||
|
*
|
||
|
* :: name
|
||
|
* le nom de propriété ou de clé à initialiser en réponse à l'utilisation de
|
||
|
* l'option.
|
||
|
*
|
||
|
* Si la destination est un objet, alors c'est une propriété qui est mise à
|
||
|
* jour, sinon ça doit être un tableau, et c'est une clé qui est mise à jour.
|
||
|
*
|
||
|
* l'utilisation de "property" ou "key" forcent respectivement la mise à jour
|
||
|
* d'une propriété ou d'une clé
|
||
|
*
|
||
|
* "property" est prioritaire par rapport à "key", qui est lui même prioritaire
|
||
|
* par rapport à "name". "name" est automatiquement mis à jour en fonction de la
|
||
|
* valeur de "property" ou "key"
|
||
|
*
|
||
|
* :: ensure_array
|
||
|
* force la destination à être un tableau. C'est le cas par défaut pour le reste
|
||
|
* des arguments et/ou si le nombre d'argument possible est plus grand que 1
|
||
|
*
|
||
|
* XXX il manque les fonctionnalités suivantes:
|
||
|
* - gestion des commandes
|
||
|
* - génération des scripts d'auto-complétion bash
|
||
|
*/
|
||
|
class ArgsParser {
|
||
|
const KIND_OPTION = "option";
|
||
|
const KIND_COMMAND = "command";
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
# méthodes surchargeables
|
||
|
|
||
|
protected function invalidDef(string $msg): ArgsException {
|
||
|
return new ArgsException("invalid definition: $msg");
|
||
|
}
|
||
|
|
||
|
protected function notEnoughArgs(int $needed, ?string $what=null): ArgsException {
|
||
|
if ($what !== null) $what = " for $what";
|
||
|
return new ArgsException("needs $needed more argument(s)$what");
|
||
|
}
|
||
|
|
||
|
protected function tooManyArgs(int $count, int $expected, ?string $what=null): ArgsException {
|
||
|
if ($what !== null) $what = " for $what";
|
||
|
return new ArgsException("too many arguments$what (expected $expected, got $count)");
|
||
|
}
|
||
|
|
||
|
protected function invalidArg(string $arg, string $kind): ArgsException {
|
||
|
return new ArgsException("$arg: invalid $kind");
|
||
|
}
|
||
|
|
||
|
protected function ambiguousArg(string $arg, string $kind, array $candidates): ArgsException {
|
||
|
$candidates = implode(", ", $candidates);
|
||
|
return new ArgsException("$arg: ambiguous $kind (possible candidates are $candidates)");
|
||
|
}
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
/**
|
||
|
* Construire un parser avec les définitions spécifiées. La méthode reset()
|
||
|
* est appelée automatiquement.
|
||
|
*
|
||
|
* @throws ArgsException en cas d'erreur de syntaxe
|
||
|
*/
|
||
|
function __construct(array $defs) {
|
||
|
[$meta, $sections, $sdefs, $ldefs, $cdefs, $rdef] = $this->parseDefs($defs);
|
||
|
$defaultDefs = [$defs, $meta, $sections, $sdefs, $ldefs, $cdefs, $rdef];
|
||
|
$this->defaults = $defaultDefs;
|
||
|
$this->defs = $defs;
|
||
|
$this->meta = $meta;
|
||
|
$this->sections = $sections;
|
||
|
$this->sdefs = $sdefs;
|
||
|
$this->ldefs = $ldefs;
|
||
|
$this->cdefs = $cdefs;
|
||
|
$this->rdef = $rdef;
|
||
|
}
|
||
|
|
||
|
/** @var array définitions par défaut */
|
||
|
private $defaults;
|
||
|
|
||
|
/** @var array définitions courantes */
|
||
|
private $defs;
|
||
|
private $meta;
|
||
|
private $sections;
|
||
|
private $sdefs;
|
||
|
private $ldefs;
|
||
|
private $cdefs;
|
||
|
private $rdef;
|
||
|
|
||
|
/** @var object|array objet destination */
|
||
|
protected $dest;
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
/**
|
||
|
* Analyser la ligne de commande et initialiser l'objet ou le tableau $dest
|
||
|
* selon les règles formulées par les définitions
|
||
|
*
|
||
|
* @throws ArgsException
|
||
|
*/
|
||
|
function parse(&$dest, array $args=null): void {
|
||
|
if ($args === null) {
|
||
|
global $argv;
|
||
|
$args = array_slice($argv, 1);
|
||
|
}
|
||
|
if ($dest === null) $dest = new stdClass();
|
||
|
$this->setDest($dest);
|
||
|
$args = $this->normArgs($args);
|
||
|
$this->parseArgs($args);
|
||
|
$this->unsetDest();
|
||
|
}
|
||
|
|
||
|
function setDest(&$dest): void {
|
||
|
$this->dest =& $dest;
|
||
|
}
|
||
|
|
||
|
function unsetDest(): void {
|
||
|
unset($this->dest);
|
||
|
}
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
/**
|
||
|
* Modifier l'ensemble courant des arguments valides à ceux définis par la
|
||
|
* commande spécifiée. Utiliser null pour reprendre l'analyse avec les
|
||
|
* arguments par défaut
|
||
|
*/
|
||
|
function select(?string $command): void {
|
||
|
if ($command === null) {
|
||
|
[$this->defs, $this->meta, $this->sections, $this->sdefs, $this->ldefs, $this->cdefs, $this->rdef,
|
||
|
] = $this->defaults;
|
||
|
} else {
|
||
|
$dyncdefs = $this->cdefs;
|
||
|
$this->checkDyncdefs($this->meta["dynamic_command"], $command, $dyncdefs);
|
||
|
$def = A::get($dyncdefs, $command);
|
||
|
if ($def === null) throw $this->invalidArg($command, self::KIND_COMMAND);
|
||
|
$this->defs = $def["cmd_args"];
|
||
|
[$this->meta, $this->sections, $this->sdefs, $this->ldefs, $this->cdefs, $this->rdef,
|
||
|
] = $def["cmd_defs"];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
/**
|
||
|
* analyser les définitions et retourner les tableaux suivants:
|
||
|
* - $meta, les méta-informations sur la liste des définitions
|
||
|
* - $sections, une liste de sections avec leurs définitions normalisées. la
|
||
|
* première section est la section par défaut
|
||
|
* - sdefs, le tableau des options courtes
|
||
|
* - ldefs, le tableau des options longues
|
||
|
* - cdefs, le tableau des commandes
|
||
|
* - rdef, la définition des arguments restants
|
||
|
*
|
||
|
* si $autohelp, rajouter automatiquement une option --help qui affiche
|
||
|
* l'aide si elle n'existe pas déjà
|
||
|
*
|
||
|
* si $autoremains, rajouter automatiquement la gestion des arguments
|
||
|
* restants en les plaçant dans la propriété de destination "args", si cette
|
||
|
* gestion n'est pas déjà définie.
|
||
|
*
|
||
|
* @throws ArgsException en cas d'erreur de syntaxe
|
||
|
*/
|
||
|
function parseDefs(array $defs, bool $autohelp=true, bool $autoremains=true): array {
|
||
|
[$defs, $meta] = A::split_assoc($defs);
|
||
|
|
||
|
md::ensure_schema($meta, ref_args::DEFS_SCHEMA, null, false);
|
||
|
$autohelp = A::replace_n($meta, "autohelp", $autohelp);
|
||
|
$autoremains = A::replace_n($meta, "autoremains", $autoremains);
|
||
|
$msections = A::getdel($meta, "sections");
|
||
|
|
||
|
# Merger les tableaux supplémentaires
|
||
|
$set_defaults = $meta["set_defaults"];
|
||
|
$merge_arrays = $meta["merge_arrays"];
|
||
|
$merge = $meta["merge"];
|
||
|
if ($merge_arrays !== null || $merge !== null) {
|
||
|
if ($merge_arrays === null) $merge_arrays = [];
|
||
|
if ($merge !== null) $merge_arrays[] = $merge;
|
||
|
}
|
||
|
A::ensure_array($defs);
|
||
|
if ($set_defaults !== null || $merge_arrays !== null) {
|
||
|
if ($set_defaults !== null) {
|
||
|
# set_defaults
|
||
|
[$defs_defaults, $meta_defaults] = A::split_assoc($set_defaults);
|
||
|
md::ensure_schema($meta_defaults, ref_args::DEFS_SCHEMA, null, false);
|
||
|
$msections_defaults = A::getdel($meta_defaults, "sections");
|
||
|
A::update_n($defs, $defs_defaults);
|
||
|
A::update_n($meta, $meta_defaults);
|
||
|
A::update_n($msections, $msections_defaults);
|
||
|
}
|
||
|
if ($merge_arrays !== null) {
|
||
|
$cur_defs = $defs; $cur_meta = $meta; $cur_msections = $msections;
|
||
|
$defs = []; $meta = []; $msections = [];
|
||
|
foreach ($merge_arrays as $merge) {
|
||
|
[$merge_defs, $merge_meta] = A::split_assoc($merge);
|
||
|
md::ensure_schema($merge_meta, ref_args::DEFS_SCHEMA, null, false);
|
||
|
$merge_msections = A::getdel($merge_meta, "sections");
|
||
|
A::merge_nn($defs, $merge_defs);
|
||
|
A::merge_nn($meta, $merge_meta);
|
||
|
A::merge_nn($msections, $merge_msections);
|
||
|
}
|
||
|
A::merge($defs, $cur_defs);
|
||
|
A::merge($meta, $cur_meta);
|
||
|
A::merge($msections, $cur_msections);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
global $argv;
|
||
|
A::replace_n($meta, "name", basename($argv[0]));
|
||
|
|
||
|
$dynamic_command = $meta["dynamic_command"];
|
||
|
if ($dynamic_command !== null) {
|
||
|
if (is_string($dynamic_command) && class_exists($dynamic_command)) {
|
||
|
$dynamic_command = func::cons($dynamic_command);
|
||
|
} elseif (func::is_static($dynamic_command) || func::is_method($dynamic_command)) {
|
||
|
$dynamic_command = new DynamicCommandMethod($dynamic_command);
|
||
|
} else {
|
||
|
throw $this->invalidDef("dynamic_command");
|
||
|
}
|
||
|
$meta["dynamic_command"] = $dynamic_command;
|
||
|
}
|
||
|
|
||
|
$argsname = $meta["argsname"];
|
||
|
$argsproperty = $meta["argsproperty"];
|
||
|
$argskey = $meta["argskey"];
|
||
|
if ($argsname === null && $argsproperty === null && $argskey === null) $argsname = "args";
|
||
|
elseif ($argsname === null && $argsproperty !==null) $argsname = $argsproperty;
|
||
|
elseif ($argsname === null && $argskey !==null) $argsname = $argskey;
|
||
|
$meta["argsname"] = $argsname;
|
||
|
|
||
|
$commandname = $meta["commandname"];
|
||
|
$commandproperty = $meta["commandproperty"];
|
||
|
$commandkey = $meta["commandkey"];
|
||
|
if ($commandname === null && $commandproperty === null && $commandkey === null) $commandname = "command";
|
||
|
elseif ($commandname === null && $commandproperty !==null) $commandname = $commandproperty;
|
||
|
elseif ($commandname === null && $commandkey !==null) $commandname = $commandkey;
|
||
|
$meta["commandname"] = $commandname;
|
||
|
|
||
|
# analyser les options
|
||
|
$sections = [];
|
||
|
$sdefs = [];
|
||
|
$ldefs = [];
|
||
|
$cdefs = [];
|
||
|
$rdef = null;
|
||
|
$have_remains = false;
|
||
|
$have_help = false;
|
||
|
|
||
|
if ($msections !== null) {
|
||
|
foreach ($msections as $msection) {
|
||
|
[$mdefs, $msection] = A::split_assoc($msection);
|
||
|
$sections[] = $this->buildSection($msection, $mdefs, $meta, $sdefs, $ldefs, $cdefs, $rdef, $have_remains, $have_help);
|
||
|
}
|
||
|
}
|
||
|
# calculer la section par défaut à la fin, pour qu'elle puisse surcharger
|
||
|
# des options définies précédemment. par contre, il faut la mettre au début
|
||
|
# du tableau $sections
|
||
|
array_unshift($sections, $this->buildSection([], $defs, $meta, $sdefs, $ldefs, $cdefs, $rdef, $have_remains, $have_help));
|
||
|
|
||
|
if (!$have_remains && $autoremains) {
|
||
|
$def = [
|
||
|
"args" => [null],
|
||
|
"action" => "--meta",
|
||
|
"name" => "args",
|
||
|
];
|
||
|
$this->parseDef($def, $sdefs, $ldefs, $cdefs, $rdef);
|
||
|
}
|
||
|
if (!$have_help && $autohelp) {
|
||
|
$self = $this;
|
||
|
$def = [
|
||
|
"--help",
|
||
|
"action" => function() use ($self, $meta, $sections) {
|
||
|
$self->printHelp($meta, $sections);
|
||
|
exit(0);
|
||
|
},
|
||
|
"help" => "Afficher l'aide",
|
||
|
];
|
||
|
$this->parseDef($def, $sdefs, $ldefs, $cdefs, $rdef);
|
||
|
$def = [
|
||
|
"--help-all",
|
||
|
"action" => function() use ($self, $meta, $sections) {
|
||
|
$self->printHelp($meta, $sections, true);
|
||
|
exit(0);
|
||
|
},
|
||
|
"help" => "Afficher l'aide complète",
|
||
|
];
|
||
|
$this->parseDef($def, $sdefs, $ldefs, $cdefs, $rdef);
|
||
|
}
|
||
|
return [$meta, $sections, $sdefs, $ldefs, $cdefs, $rdef];
|
||
|
}
|
||
|
|
||
|
private function buildSection(
|
||
|
?array $section, ?array $defs,
|
||
|
?array $meta, array &$sdefs, array &$ldefs, array &$cdefs, ?array &$rdef,
|
||
|
bool &$have_remains, bool &$have_help
|
||
|
): array {
|
||
|
A::ensure_array($section);
|
||
|
$section["defs"] = A::with($defs);
|
||
|
md::ensure_schema($section, ref_args::SECTION_SCHEMA, null, false);
|
||
|
foreach ($section["defs"] as &$def) {
|
||
|
if ($this->isGroup($def)) {
|
||
|
# groupe d'option
|
||
|
$count = count($def);
|
||
|
for ($i = 1; $i < $count; $i++) {
|
||
|
$this->parseDef($def[$i], $sdefs, $ldefs, $cdefs, $rdef, $have_remains, $have_help);
|
||
|
}
|
||
|
} else {
|
||
|
# option simple
|
||
|
$this->parseDef($def, $sdefs, $ldefs, $cdefs, $rdef, $have_remains, $have_help);
|
||
|
}
|
||
|
}; unset($def);
|
||
|
return $section;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* vérifier si $def est de la forme ["group", ...$defs]
|
||
|
*
|
||
|
* au niveau de l'affichage de l'aide, toutes les options d'un même groupe
|
||
|
* sont affichées ensemble, avec comme aide le texte du premier élément du
|
||
|
* groupe
|
||
|
*/
|
||
|
private function isGroup(array $def): bool {
|
||
|
# $def ne doit pas être associatif
|
||
|
$assoc = A::split_assoc($def)[1];
|
||
|
if ($assoc !== null) return false;
|
||
|
# il faut au moins 2 valeurs et la première valeur doit être "group"
|
||
|
if (count($def) < 2 || $def[0] !== "group") return false;
|
||
|
# toutes les valeurs suivantes doivent être des tableaux
|
||
|
$count = count($def);
|
||
|
for ($i = 1; $i < $count; $i++) {
|
||
|
if (!is_array($def[$i])) return false;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
private function parseDef(
|
||
|
array &$def,
|
||
|
array &$sdefs, array &$ldefs, array &$cdefs, ?array &$rdef,
|
||
|
bool &$have_remains=false, bool &$have_help=false
|
||
|
): void {
|
||
|
# Merger les tableaux supplémentaires
|
||
|
$set_defaults = A::getdel($def, "set_defaults");
|
||
|
$merge_arrays = A::getdel($def, "merge_arrays");
|
||
|
$merge = A::getdel($def, "merge");
|
||
|
if ($merge_arrays !== null || $merge !== null) {
|
||
|
if ($merge_arrays === null) $merge_arrays = [];
|
||
|
if ($merge !== null) $merge_arrays[] = $merge;
|
||
|
}
|
||
|
if ($set_defaults !== null) {
|
||
|
A::update_n($def, $set_defaults);
|
||
|
}
|
||
|
if ($merge_arrays !== null) {
|
||
|
foreach ($merge_arrays as $merge) {
|
||
|
A::merge_nn($def, $merge);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
[$seq, $assoc] = A::split_assoc($def);
|
||
|
md::ensure_schema($assoc, ref_args::DEF_SCHEMA, null, false);
|
||
|
|
||
|
$args_default = null;
|
||
|
if ($seq === null) {
|
||
|
$is_remains = $have_remains = true;
|
||
|
} else {
|
||
|
$is_remains = false;
|
||
|
foreach ($seq as &$opt) {
|
||
|
if (str::_starts_with("--", $opt) && str::_ends_with("=", $opt)) {
|
||
|
$opt = str::without_suffix("=", $opt);
|
||
|
if ($args_default === null) $args_default = true;
|
||
|
} elseif (str::_starts_with("-", $opt) && str::_ends_with("::", $opt)) {
|
||
|
$opt = str::without_suffix("::", $opt);
|
||
|
if ($args_default === null) $args_default = [["value"]];
|
||
|
} elseif (str::_starts_with("-", $opt) && str::_ends_with(":", $opt)) {
|
||
|
$opt = str::without_suffix(":", $opt);
|
||
|
if ($args_default === null) $args_default = true;
|
||
|
}
|
||
|
}; unset($opt);
|
||
|
if (in_array("--help", $seq)) $have_help = true;
|
||
|
}
|
||
|
# $desc est une description de l'argument présentement défini, utilisée dans
|
||
|
# les levées d'exceptions
|
||
|
if ($is_remains) $desc = "remaining args";
|
||
|
else $desc = $def[0];
|
||
|
|
||
|
## arguments que cette définition peut prendre
|
||
|
$args = $assoc["args"];
|
||
|
if ($args === null) $args = $assoc["arg"];
|
||
|
if ($args === null) $args = $args_default;
|
||
|
if ($args === true) $args = 1;
|
||
|
if (is_int($args)) $args = array_fill(0, $args, "value");
|
||
|
$args = A::with($args);
|
||
|
$have_args = $args !== [];
|
||
|
$arg_values = null;
|
||
|
$min_args = 0;
|
||
|
$max_args = 0;
|
||
|
$argsdesc = [];
|
||
|
if ($have_args) {
|
||
|
$reqs = [];
|
||
|
$opts = [];
|
||
|
$have_opt = false;
|
||
|
$have_null = false;
|
||
|
|
||
|
$nb_seen = 0;
|
||
|
$opt_args = null;
|
||
|
foreach ($args as $arg) {
|
||
|
$nb_seen++;
|
||
|
if (is_string($arg)) {
|
||
|
$reqs[] = $arg;
|
||
|
$argsdesc[] = strtoupper($arg);
|
||
|
} elseif (is_array($arg)) {
|
||
|
$opt_args = $arg;
|
||
|
break;
|
||
|
} elseif ($arg === null) {
|
||
|
$have_null = true;
|
||
|
break;
|
||
|
} else {
|
||
|
throw $this->invalidDef("$desc: arg must be string, array or null");
|
||
|
}
|
||
|
}
|
||
|
if ($nb_seen != count($args)) {
|
||
|
throw $this->invalidDef("$desc: invalid args format");
|
||
|
}
|
||
|
|
||
|
$opt_argsdesc = [];
|
||
|
$lastarg = strtoupper(ref_args::ARGS_ALLOWED_VALUES[0]);
|
||
|
if ($opt_args !== null) {
|
||
|
$nb_seen = 0;
|
||
|
foreach ($opt_args as $arg) {
|
||
|
$nb_seen++;
|
||
|
if (is_string($arg)) {
|
||
|
$have_opt = true;
|
||
|
$opts[] = $arg;
|
||
|
$lastarg = strtoupper($arg);
|
||
|
$opt_argsdesc[] = "$lastarg";
|
||
|
} elseif ($arg === null) {
|
||
|
$have_null = true;
|
||
|
break;
|
||
|
} else {
|
||
|
throw $this->invalidDef("$desc: optional arg must be string or null");
|
||
|
}
|
||
|
}
|
||
|
if ($nb_seen != count($opt_args)) {
|
||
|
throw $this->invalidDef("$desc: invalid args format");
|
||
|
}
|
||
|
if (!$have_opt) $have_null = true;
|
||
|
}
|
||
|
if ($have_null) $opt_argsdesc[] = "${lastarg}s...";
|
||
|
if ($opt_argsdesc) {
|
||
|
$argsdesc[] = "[".implode(" ", $opt_argsdesc)."]";
|
||
|
}
|
||
|
|
||
|
$min_args = count($reqs);
|
||
|
$nb_opts = count($opts);
|
||
|
if ($have_null) $max_args = PHP_INT_MAX;
|
||
|
else $max_args = $min_args + $nb_opts;
|
||
|
|
||
|
$arg_values = array_merge($reqs, $opts);
|
||
|
foreach ($arg_values as $arg_value) {
|
||
|
if (!in_array($arg_value, ref_args::ARGS_ALLOWED_VALUES)) {
|
||
|
throw $this->invalidDef("$desc: arg value must be in ".implode(", ", ref_args::ARGS_ALLOWED_VALUES));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
$argsdesc = implode(" ", $argsdesc);
|
||
|
$argsdesc = A::replace_n($assoc, "argsdesc", $argsdesc);
|
||
|
|
||
|
# ensure_array vaut true par défaut pour remains ou si le nombre maximum
|
||
|
# d'arguments est plus grand que 1
|
||
|
$ensure_array = $is_remains || $max_args > 1? true: false;
|
||
|
$ensure_array = A::replace_n($assoc, "ensure_array", $ensure_array);
|
||
|
# normaliser l'action
|
||
|
$action = $assoc["action"];
|
||
|
$defaultAction = $action === null;
|
||
|
if ($defaultAction) {
|
||
|
if ($have_args) {
|
||
|
$action = "--set";
|
||
|
} else {
|
||
|
if ($assoc["value"] !== null) $action = "--set";
|
||
|
elseif ($assoc["inverse"]) $action = "--dec";
|
||
|
else $action = "--inc";
|
||
|
}
|
||
|
}
|
||
|
$assoc["action"] = $action;
|
||
|
$is_func = !is_string($action) || substr($action, 0, 2) != "--";
|
||
|
# normaliser le nom
|
||
|
$name = $assoc["name"];
|
||
|
$property = $assoc["property"];
|
||
|
$key = $assoc["key"];
|
||
|
if (!$is_func && !$is_remains && $name === null && $property === null && $key === null) {
|
||
|
# si on ne précise pas le nom de la propriété, la dériver à partir du
|
||
|
# nom de l'option la plus longue
|
||
|
$longest = null;
|
||
|
$maxlen = 0;
|
||
|
foreach ($seq as $item) {
|
||
|
if (substr($item, 0, 2) == "--") {
|
||
|
$long = substr($item, 2);
|
||
|
$len = strlen($long);
|
||
|
if ($len > $maxlen) {
|
||
|
$longest = $long;
|
||
|
$maxlen = $len;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ($longest === null) {
|
||
|
# si pas d'option longue, essayer avec nom de commande
|
||
|
$maxlen = 0;
|
||
|
foreach ($seq as $item) {
|
||
|
if (substr($item, 0, 1) != "-") {
|
||
|
$long = $item;
|
||
|
$len = strlen($long);
|
||
|
if ($len > $maxlen) {
|
||
|
$longest = $long;
|
||
|
$maxlen = $len;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ($longest === null) {
|
||
|
# sinon avec option courte
|
||
|
$maxlen = 0;
|
||
|
foreach ($seq as $item) {
|
||
|
if (substr($item, 0, 1) == "-") {
|
||
|
$long = substr($item, 1);
|
||
|
$len = strlen($long);
|
||
|
if ($len > $maxlen) {
|
||
|
$longest = $long;
|
||
|
$maxlen = $len;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if ($longest !== null) {
|
||
|
$longest = preg_replace('/[^A-Za-z0-9]+/', "_", $longest);
|
||
|
if (preg_match('/^[0-9]/', $longest)) {
|
||
|
# le nom de la propriété ne doit pas commencer par un chiffre
|
||
|
$longest = "p$longest";
|
||
|
}
|
||
|
$name = $longest;
|
||
|
}
|
||
|
} elseif ($name === null && $property !== null) {
|
||
|
$name = $property;
|
||
|
} elseif ($name === null && $key !== null) {
|
||
|
$name = $key;
|
||
|
}
|
||
|
$assoc["name"] = $name;
|
||
|
$assoc["property"] = $property;
|
||
|
$assoc["key"] = $key;
|
||
|
|
||
|
$cmd_args = $assoc["cmd_args"];
|
||
|
if ($cmd_args !== null) {
|
||
|
A::replace_n($cmd_args, "purpose", $assoc["help"]);
|
||
|
$assoc["cmd_defs"] = $this->parseDefs($cmd_args);
|
||
|
}
|
||
|
|
||
|
$basedef = array_merge($assoc, [
|
||
|
"is_option" => false, # est-ce une option?
|
||
|
"options" => null, # options valides
|
||
|
"option" => null, # valeur de cette option
|
||
|
"is_short" => false, # est-ce une option courte?
|
||
|
"is_long" => false, # est-ce une option longue?
|
||
|
"is_command" => false, # est-ce une commande?
|
||
|
"commands" => null, # commandes valides
|
||
|
"command" => null, # valeur de cette commande
|
||
|
"is_remains" => false, # est-ce le reste des arguments de la ligne de commande?
|
||
|
"have_args" => $have_args, # cette définition (option/commande/reste) prend-elle des arguments?
|
||
|
"arg_types" => $arg_values, # type des arguments parmi value, path, host
|
||
|
"min_args" => $min_args, # nombre minimum d'arguments (nombre d'arguments requis)
|
||
|
"max_args" => $max_args, # nombre maximum d'arguments (différent si certains arguments sont optionnels)
|
||
|
]);
|
||
|
if ($is_remains) {
|
||
|
$def = $rdef = array_merge($basedef, [
|
||
|
"is_remains" => true,
|
||
|
]);
|
||
|
} else {
|
||
|
$kind = $assoc["kind"];
|
||
|
switch ($kind) {
|
||
|
case "o":
|
||
|
case "opt":
|
||
|
case "option":
|
||
|
$kind = self::KIND_OPTION;
|
||
|
break;
|
||
|
case "c":
|
||
|
case "cmd":
|
||
|
case "command":
|
||
|
$kind = self::KIND_COMMAND;
|
||
|
break;
|
||
|
default:
|
||
|
$kind = null; # autodétecter sur la base du premier
|
||
|
}
|
||
|
$bdef = array_merge($seq, $basedef);
|
||
|
$first = true;
|
||
|
$first_item = null;
|
||
|
foreach ($seq as $item) {
|
||
|
if ($item == "--") {
|
||
|
$this->invalidDef("$item: is reserved");
|
||
|
}
|
||
|
if ($kind === null) {
|
||
|
if (substr($item, 0, 1) == "-") $kind = self::KIND_OPTION;
|
||
|
else $kind = self::KIND_COMMAND;
|
||
|
}
|
||
|
if ($first) $first_item = $item;
|
||
|
if ($kind === self::KIND_OPTION) {
|
||
|
if (substr($item, 0, 1) == "-" && strlen($item) == 2) {
|
||
|
$sopt = substr($item, 1);
|
||
|
$sdefs[$sopt] = $sdef = array_merge($bdef, [
|
||
|
"is_option" => true,
|
||
|
"options" => $seq,
|
||
|
"option" => $item,
|
||
|
"is_short" => true,
|
||
|
]);
|
||
|
if ($first) $def = $sdef;
|
||
|
} else {
|
||
|
$lopt = $item;
|
||
|
$ldefs[$lopt] = $ldef = array_merge($bdef, [
|
||
|
"is_option" => true,
|
||
|
"options" => $seq,
|
||
|
"option" => $item,
|
||
|
"is_long" => true,
|
||
|
]);
|
||
|
if ($first) $def = $ldef;
|
||
|
}
|
||
|
} elseif ($kind == "command") {
|
||
|
$cmd = $item;
|
||
|
$cdefs[$cmd] = $cdef = array_merge($bdef
|
||
|
, $defaultAction? [
|
||
|
"action" => "--meta",
|
||
|
"name" => "command",
|
||
|
"value" => $first_item,
|
||
|
]: []
|
||
|
, [
|
||
|
"is_command" => true,
|
||
|
"commands" => $seq,
|
||
|
"command" => $item,
|
||
|
]);
|
||
|
if ($first) $def = $cdef;
|
||
|
}
|
||
|
$first = false;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
private function checkDyncdefs(?IDynamicCommand $dc, string $command, ?array &$dyncdefs, bool $virtual=false): ?string {
|
||
|
if ($dc === null) return null;
|
||
|
if ($dc instanceof DynamicCommandMethod) $dc->setDest($this->dest);
|
||
|
$dcdefs = $dc->getCommandDefs($command, $virtual);
|
||
|
if ($dcdefs === null) return null;
|
||
|
|
||
|
[$dcmeta, $dcsections, $dcsdefs, $dcldefs, $dccdefs, $dcrdef] = $this->parseDefs($dcdefs);
|
||
|
foreach ($dccdefs as $dcname => $dccdef) {
|
||
|
$dyncdefs[$dcname] = $dccdef;
|
||
|
}
|
||
|
return A::has($dyncdefs, $command)? "cmd": null;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* normaliser les arguments de $args: détacher les options les unes des
|
||
|
* autres et réordonner les arguments
|
||
|
*
|
||
|
* @throws ArgsException
|
||
|
*/
|
||
|
function normArgs(array $args, bool $reset=true): array {
|
||
|
if ($reset) $this->select(null);
|
||
|
$meta = $this->meta;
|
||
|
$sdefs = $this->sdefs;
|
||
|
$ldefs = $this->ldefs;
|
||
|
$cdefs = $this->cdefs;
|
||
|
|
||
|
$i = 0;
|
||
|
$max = count($args);
|
||
|
$norm_args = [];
|
||
|
$first_group = true;
|
||
|
while (true) {
|
||
|
$parse_opts = true;
|
||
|
$options = [];
|
||
|
$remains = [];
|
||
|
while ($i < $max) {
|
||
|
$arg = $args[$i++];
|
||
|
if ($arg == "//") {
|
||
|
# changement de groupe d'option, remettre à zéro l'analyseur
|
||
|
$this->select(null);
|
||
|
$meta = $this->meta;
|
||
|
$sdefs = $this->sdefs;
|
||
|
$ldefs = $this->ldefs;
|
||
|
$cdefs = $this->cdefs;
|
||
|
break;
|
||
|
|
||
|
} elseif (!$parse_opts) {
|
||
|
# le reste de la ligne ne contient que des arguments
|
||
|
$remains[] = $arg;
|
||
|
continue;
|
||
|
|
||
|
} elseif ($arg == "--") {
|
||
|
# fin des options
|
||
|
$parse_opts = false;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$pos = strpos($arg, "=");
|
||
|
$name = $pos !== false? substr($arg, 0, $pos): $arg;
|
||
|
$dyncdefs = $cdefs;
|
||
|
$kind = $this->checkDyncdefs($meta["dynamic_command"], $arg, $dyncdefs);
|
||
|
if ($kind === null) {
|
||
|
if (A::has($ldefs, $name)) $kind = "lopt";
|
||
|
elseif (A::has($dyncdefs, $name)) $kind = "cmd";
|
||
|
elseif (substr($arg, 0, 2) == "--") $kind = "lopt";
|
||
|
elseif (substr($arg, 0, 1) == "-") $kind = "sopt";
|
||
|
elseif (A::has($dyncdefs, $arg)) $kind = "cmd";
|
||
|
}
|
||
|
|
||
|
if ($kind == "lopt") {
|
||
|
#######################################################################
|
||
|
# option longue
|
||
|
$pos = strpos($arg, "=");
|
||
|
if ($pos !== false) {
|
||
|
# option avec valeur
|
||
|
$name = substr($arg, 0, $pos);
|
||
|
$value = substr($arg, $pos + 1);
|
||
|
} else {
|
||
|
# option sans valeur
|
||
|
$name = $arg;
|
||
|
$value = null;
|
||
|
}
|
||
|
$def = A::get($ldefs, $name);
|
||
|
if ($def === null) {
|
||
|
# chercher une correspondance
|
||
|
$length = strlen($name);
|
||
|
$candidates = [];
|
||
|
foreach (array_keys($ldefs) as $lopt) {
|
||
|
if (substr($lopt, 0, $length) == $name) {
|
||
|
$candidates[] = $lopt;
|
||
|
}
|
||
|
}
|
||
|
switch (count($candidates)) {
|
||
|
case 0:
|
||
|
throw $this->invalidArg($arg, self::KIND_OPTION);
|
||
|
case 1:
|
||
|
$name = $candidates[0];
|
||
|
break;
|
||
|
default:
|
||
|
throw $this->ambiguousArg($name, self::KIND_OPTION, $candidates);
|
||
|
}
|
||
|
$def = A::get($ldefs, $name);
|
||
|
}
|
||
|
|
||
|
$option = $def["option"];
|
||
|
if ($def["have_args"]) {
|
||
|
$min_args = $def["min_args"];
|
||
|
$max_args = $def["max_args"];
|
||
|
$values = [];
|
||
|
if ($value !== null) {
|
||
|
$values[] = $value;
|
||
|
$offset = 1;
|
||
|
} elseif ($min_args == 0) {
|
||
|
# cas particulier: la première valeur doit être collée à l'option si $max_args==1
|
||
|
$offset = $max_args == 1? 1: 0;
|
||
|
} else {
|
||
|
$offset = 0;
|
||
|
}
|
||
|
$this->checkEnoughArgs("option $option",
|
||
|
self::consume_args($args, $i, $values, $offset, $min_args, $max_args, true));
|
||
|
if ($min_args == 0 && $max_args == 1) {
|
||
|
# cas particulier: la première valeur doit être collée à l'option
|
||
|
if (count($values) > 0) {
|
||
|
$options[] = "$option=$values[0]";
|
||
|
$values = array_slice($values, 1);
|
||
|
} else {
|
||
|
$options[] = $option;
|
||
|
}
|
||
|
} else {
|
||
|
$options[] = $option;
|
||
|
}
|
||
|
$options = array_merge($options, $values);
|
||
|
} else {
|
||
|
$options[] = $option;
|
||
|
}
|
||
|
|
||
|
} elseif ($kind == "sopt") {
|
||
|
#######################################################################
|
||
|
# option courte
|
||
|
$pos = 1;
|
||
|
$length = strlen($arg);
|
||
|
while ($pos < $length) {
|
||
|
$def = A::get($sdefs, substr($arg, $pos, 1));
|
||
|
if ($def === null) throw $this->invalidArg($arg, self::KIND_OPTION);
|
||
|
|
||
|
$option = $def["option"];
|
||
|
if ($def["have_args"]) {
|
||
|
$min_args = $def["min_args"];
|
||
|
$max_args = $def["max_args"];
|
||
|
$values = [];
|
||
|
if ($length > $pos + 1) {
|
||
|
# option avec valeur
|
||
|
$values[] = substr($arg, $pos + 1);
|
||
|
$offset = 1;
|
||
|
$pos = $length;
|
||
|
} elseif ($min_args == 0) {
|
||
|
# cas particulier: la première valeur doit être collée à l'option si $max_args==1
|
||
|
$offset = $max_args == 1? 1: 0;
|
||
|
} else {
|
||
|
# option sans valeur
|
||
|
$offset = 0;
|
||
|
}
|
||
|
$this->checkEnoughArgs("option $option",
|
||
|
self::consume_args($args, $i, $values, $offset, $min_args, $max_args, true));
|
||
|
if ($min_args == 0 && $max_args == 1) {
|
||
|
# cas particulier: la première valeur doit être collée à l'option
|
||
|
if (count($values) > 0) {
|
||
|
$options[] = "$option$values[0]";
|
||
|
$values = array_slice($values, 1);
|
||
|
} else {
|
||
|
$options[] = $option;
|
||
|
}
|
||
|
} else {
|
||
|
$options[] = $option;
|
||
|
}
|
||
|
$options = array_merge($options, $values);
|
||
|
} else {
|
||
|
$options[] = $option;
|
||
|
}
|
||
|
$pos++;
|
||
|
}
|
||
|
|
||
|
} elseif ($kind == "cmd") {
|
||
|
#######################################################################
|
||
|
# commande
|
||
|
$pos = strpos($arg, "=");
|
||
|
if ($pos !== false) {
|
||
|
# option avec valeur
|
||
|
$name = substr($arg, 0, $pos);
|
||
|
$value = substr($arg, $pos + 1);
|
||
|
} else {
|
||
|
# option sans valeur
|
||
|
$name = $arg;
|
||
|
$value = null;
|
||
|
}
|
||
|
$def = A::get($dyncdefs, $name);
|
||
|
if ($def === null) throw $this->invalidArg($arg, self::KIND_COMMAND);
|
||
|
|
||
|
$command = $def["command"];
|
||
|
if ($def["have_args"]) {
|
||
|
$min_args = $def["min_args"];
|
||
|
$max_args = $def["max_args"];
|
||
|
$values = [];
|
||
|
if ($value !== null) {
|
||
|
$values[] = $value;
|
||
|
$offset = 1;
|
||
|
} elseif ($min_args == 0) {
|
||
|
# cas particulier: la première valeur doit être collée à la commande si $max_args==1
|
||
|
$offset = $max_args == 1? 1: 0;
|
||
|
} else {
|
||
|
$offset = 0;
|
||
|
}
|
||
|
$this->checkEnoughArgs("command $command",
|
||
|
self::consume_args($args, $i, $values, $offset, $min_args, $max_args, true));
|
||
|
if ($min_args == 0 && $max_args == 1) {
|
||
|
# cas particulier: la première valeur doit être collée à l'option
|
||
|
if (count($values) > 0) {
|
||
|
$options[] = "$command=$values[0]";
|
||
|
$values = array_slice($values, 1);
|
||
|
} else {
|
||
|
$options[] = $command;
|
||
|
}
|
||
|
} else {
|
||
|
$options[] = $command;
|
||
|
}
|
||
|
$options = array_merge($options, $values);
|
||
|
} else {
|
||
|
$options[] = $command;
|
||
|
}
|
||
|
$this->select($command);
|
||
|
$meta = $this->meta;
|
||
|
$sdefs = $this->sdefs;
|
||
|
$ldefs = $this->ldefs;
|
||
|
$cdefs = $this->cdefs;
|
||
|
|
||
|
} else {
|
||
|
# argument
|
||
|
$remains[] = $arg;
|
||
|
}
|
||
|
}
|
||
|
if ($first_group) $first_group = false;
|
||
|
else $norm_args[] = "//";
|
||
|
$norm_args = array_merge($norm_args, $options, ["--"], $remains);
|
||
|
if ($i >= $max) break;
|
||
|
}
|
||
|
return $norm_args;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* consommer les arguments de $src en avançant l'index $srci et provisionner
|
||
|
* $dest à partir de $desti. si $desti est plus grand que 0, celà veut dire
|
||
|
* que $dest a déjà commencé à être provisionné, et qu'il faut continuer.
|
||
|
*
|
||
|
* $destmin est le nombre minimum d'arguments à consommer. $destmax est le
|
||
|
* nombre maximum d'arguments à consommer.
|
||
|
*
|
||
|
* $srci est la position de l'élément courant à consommer le cas échéant
|
||
|
* retourner le nombre d'arguments qui manquent (ou 0 si tous les arguments
|
||
|
* ont été consommés)
|
||
|
*
|
||
|
* pour les arguments optionnels, ils sont consommés tant qu'il y en a de
|
||
|
* disponible, ou jusqu'à la présence de '--'. Si $keepsep, l'argument '--'
|
||
|
* est gardé dans la liste des arguments optionnels.
|
||
|
*/
|
||
|
private static function consume_args($src, &$srci, &$dest, $desti, $destmin, $destmax, bool $keepsep): int {
|
||
|
$srcmax = count($src);
|
||
|
# arguments obligatoires
|
||
|
while ($desti < $destmin) {
|
||
|
if ($srci < $srcmax) {
|
||
|
$dest[] = $src[$srci];
|
||
|
} else {
|
||
|
# pas assez d'arguments
|
||
|
return $destmin - $desti;
|
||
|
}
|
||
|
$srci++;
|
||
|
$desti++;
|
||
|
}
|
||
|
# arguments facultatifs
|
||
|
while ($desti < $destmax && $srci < $srcmax) {
|
||
|
$opt = $src[$srci];
|
||
|
if ($opt === "--") {
|
||
|
# fin des options facultatives
|
||
|
if ($keepsep) $dest[] = $opt;
|
||
|
$srci++;
|
||
|
break;
|
||
|
}
|
||
|
$dest[] = $opt;
|
||
|
$srci++;
|
||
|
$desti++;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
private function checkEnoughArgs(?string $what, int $count) {
|
||
|
if ($count > 0) {
|
||
|
throw $this->notEnoughArgs($count, $what);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static function ensure_type(&$value, ?string $type): void {
|
||
|
switch ($type) {
|
||
|
case "?array":
|
||
|
case "array":
|
||
|
case "?array[]":
|
||
|
case "array[]":
|
||
|
if ($value !== null) {
|
||
|
$value = explode(",", $value);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
types::verifix($type, $value, $result, true);
|
||
|
}
|
||
|
|
||
|
private static function ensure_types(&$values, $types): void {
|
||
|
if (!$types) return;
|
||
|
if (is_array($values)) {
|
||
|
$types = A::with($types);
|
||
|
$index = 0;
|
||
|
foreach ($values as &$value) {
|
||
|
if (array_key_exists($index, $types)) $type = $types[$index];
|
||
|
$index++;
|
||
|
self::ensure_type($value, $type);
|
||
|
}; unset($value);
|
||
|
} else {
|
||
|
if (is_array($types)) $types = A::first($types);
|
||
|
self::ensure_type($values, $types);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
/** si $arg == \//, le remplacer par // */
|
||
|
private static function fix_arg(string $arg): string {
|
||
|
if (preg_match('/^\\\\+\/\//', $arg)) {
|
||
|
$arg = substr($arg, 1);
|
||
|
}
|
||
|
return $arg;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* analyser les arguments et mettre à jour l'objet $this->dest
|
||
|
* $args *doit* avoir été normalisé avec {@link normArgs()} d'abord
|
||
|
*
|
||
|
* @throws ArgsException
|
||
|
*/
|
||
|
function parseArgs(array $args, bool $reset=true) {
|
||
|
if ($reset) $this->select(null);
|
||
|
$meta = $this->meta;
|
||
|
$sdefs = $this->sdefs;
|
||
|
$ldefs = $this->ldefs;
|
||
|
$cdefs = $this->cdefs;
|
||
|
$rdef = $this->rdef;
|
||
|
|
||
|
$i = 0;
|
||
|
$max = count($args);
|
||
|
while (true) {
|
||
|
# d'abord traiter les options
|
||
|
$arg = null;
|
||
|
while ($i < $max) {
|
||
|
$arg = $args[$i++];
|
||
|
if ($arg == "//") {
|
||
|
# fin d'un groupe d'options
|
||
|
break;
|
||
|
} elseif ($arg == "--") {
|
||
|
# fin des options
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
$pos = strpos($arg, "=");
|
||
|
$name = $pos !== false? substr($arg, 0, $pos): $arg;
|
||
|
$dyncdefs = $cdefs;
|
||
|
$kind = $this->checkDyncdefs($meta["dynamic_command"], $arg, $dyncdefs);
|
||
|
if ($kind === null) {
|
||
|
if (A::has($ldefs, $name)) $kind = "lopt";
|
||
|
elseif (A::has($dyncdefs, $name)) $kind = "cmd";
|
||
|
elseif (substr($arg, 0, 2) == "--") $kind = "lopt";
|
||
|
elseif (substr($arg, 0, 1) == "-") $kind = "sopt";
|
||
|
elseif (A::has($dyncdefs, $arg)) $kind = "cmd";
|
||
|
}
|
||
|
|
||
|
# obtenir l'option
|
||
|
if ($kind == "lopt") {
|
||
|
#######################################################################
|
||
|
# option longue
|
||
|
$pos = strpos($arg, "=");
|
||
|
if ($pos !== false) {
|
||
|
$option = substr($arg, 0, $pos);
|
||
|
$ovalue = substr($arg, $pos + 1);
|
||
|
} else {
|
||
|
$option = $arg;
|
||
|
$ovalue = null;
|
||
|
}
|
||
|
$def = A::get($ldefs, $option);
|
||
|
$value = $def["value"];
|
||
|
if ($def["have_args"]) {
|
||
|
$min_args = $def["min_args"];
|
||
|
$max_args = $def["max_args"];
|
||
|
if ($pos !== false) {
|
||
|
# option avec valeur facultative
|
||
|
$value = [$ovalue];
|
||
|
$offset = 1;
|
||
|
} elseif ($min_args == 0 && $max_args == 1) {
|
||
|
$value = A::with($value);
|
||
|
$offset = 1;
|
||
|
} else {
|
||
|
$value = [];
|
||
|
$offset = 0;
|
||
|
}
|
||
|
self::consume_args($args, $i, $value, $offset, $min_args, $max_args, false);
|
||
|
self::ensure_types($value, $def["type"]);
|
||
|
}
|
||
|
|
||
|
} elseif ($kind == "sopt") {
|
||
|
#######################################################################
|
||
|
# option courte
|
||
|
$def = A::get($sdefs, substr($arg, 1, 1));
|
||
|
$value = $def["value"];
|
||
|
if ($def["have_args"]) {
|
||
|
$min_args = $def["min_args"];
|
||
|
$max_args = $def["max_args"];
|
||
|
$length = strlen($arg);
|
||
|
if ($length > 2) {
|
||
|
# option avec valeur facultative
|
||
|
$value = [substr($arg, 2)];
|
||
|
$offset = 1;
|
||
|
} elseif ($min_args == 0 && $max_args == 1) {
|
||
|
$value = A::with($value);
|
||
|
$offset = 1;
|
||
|
} else {
|
||
|
$value = [];
|
||
|
$offset = 0;
|
||
|
}
|
||
|
self::consume_args($args, $i, $value, $offset, $min_args, $max_args, false);
|
||
|
self::ensure_types($value, $def["type"]);
|
||
|
}
|
||
|
|
||
|
} elseif ($kind == "cmd") {
|
||
|
#######################################################################
|
||
|
# commande
|
||
|
$pos = strpos($arg, "=");
|
||
|
if ($pos !== false) {
|
||
|
$option = substr($arg, 0, $pos);
|
||
|
$ovalue = substr($arg, $pos + 1);
|
||
|
} else {
|
||
|
$option = $arg;
|
||
|
$ovalue = null;
|
||
|
}
|
||
|
$def = A::get($dyncdefs, $option);
|
||
|
$value = $def["value"];
|
||
|
if ($def["have_args"]) {
|
||
|
$min_args = $def["min_args"];
|
||
|
$max_args = $def["max_args"];
|
||
|
if ($pos !== false) {
|
||
|
# option avec valeur facultative
|
||
|
$value = [$ovalue];
|
||
|
$offset = 1;
|
||
|
} elseif ($min_args == 0 && $max_args == 1) {
|
||
|
$value = A::with($value);
|
||
|
$offset = 1;
|
||
|
} else {
|
||
|
$value = [];
|
||
|
$offset = 0;
|
||
|
}
|
||
|
self::consume_args($args, $i, $value, $offset, $min_args, $max_args, false);
|
||
|
self::ensure_types($value, $def["type"]);
|
||
|
}
|
||
|
$this->select($def["command"]);
|
||
|
$meta = $this->meta;
|
||
|
$sdefs = $this->sdefs;
|
||
|
$ldefs = $this->ldefs;
|
||
|
$cdefs = $this->cdefs;
|
||
|
$rdef = $this->rdef;
|
||
|
}
|
||
|
|
||
|
# puis traiter l'option
|
||
|
$this->doAction($value, $arg, $def);
|
||
|
}
|
||
|
|
||
|
if ($arg === "//") {
|
||
|
# fin d'un groupe d'options, réinitialiser l'analyseur
|
||
|
$this->select(null);
|
||
|
$meta = $this->meta;
|
||
|
$sdefs = $this->sdefs;
|
||
|
$ldefs = $this->ldefs;
|
||
|
$cdefs = $this->cdefs;
|
||
|
$rdef = $this->rdef;
|
||
|
$this->updateMeta("command", null);
|
||
|
|
||
|
} else {
|
||
|
# construire la liste des arguments qui restent
|
||
|
$rargs = [];
|
||
|
while ($i < $max && $args[$i] !== "//") {
|
||
|
$rargs[] = self::fix_arg($args[$i++]);
|
||
|
}
|
||
|
$ri = 0;
|
||
|
$rmax = count($rargs);
|
||
|
# puis traiter les arguments qui restent
|
||
|
if ($rdef !== null && $rdef["have_args"]) {
|
||
|
if ($rdef["max_args"] == PHP_INT_MAX) {
|
||
|
# cas particulier: si le nombre d'arguments restants est non borné,
|
||
|
# les prendre tous sans distinction ni traitement de '--'
|
||
|
$values = $rargs;
|
||
|
# mais tester tout de même s'il y a le minimum requis d'arguments
|
||
|
$this->checkEnoughArgs(null, $rdef["min_args"] - $rmax);
|
||
|
} else {
|
||
|
$values = [];
|
||
|
$this->checkEnoughArgs(null,
|
||
|
self::consume_args($rargs, $ri, $values, 0, $rdef["min_args"], $rdef["max_args"], false));
|
||
|
if ($ri <= $rmax - 1) throw $this->tooManyArgs($rmax, $ri);
|
||
|
}
|
||
|
self::ensure_types($values, $rdef["type"]);
|
||
|
$this->doAction($values, null, $rdef);
|
||
|
} else {
|
||
|
if ($ri <= $rmax - 1) throw $this->tooManyArgs($rmax, $ri);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($i >= $max) break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function updateObject(string $action, string $property, $value): void {
|
||
|
switch ($action) {
|
||
|
case "--inc":
|
||
|
oprop::inc($this->dest, $property);
|
||
|
break;
|
||
|
case "--dec":
|
||
|
oprop::dec($this->dest, $property);
|
||
|
break;
|
||
|
case "--set":
|
||
|
oprop::set($this->dest, $property, $value);
|
||
|
break;
|
||
|
case "--add":
|
||
|
oprop::append($this->dest, $property, $value);
|
||
|
break;
|
||
|
case "--adds":
|
||
|
foreach (A::with($value) as $value) {
|
||
|
oprop::append($this->dest, $property, $value);
|
||
|
}
|
||
|
break;
|
||
|
case "--merge":
|
||
|
oprop::merge($this->dest, $property, $value);
|
||
|
break;
|
||
|
case "--merges":
|
||
|
foreach (A::with($value) as $value) {
|
||
|
oprop::merge($this->dest, $property, $value);
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
throw $this->invalidDef("$action: invalid action");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function updateArray(string $action, string $key, $value): void {
|
||
|
switch ($action) {
|
||
|
case "--inc":
|
||
|
akey::inc($this->dest, $key);
|
||
|
break;
|
||
|
case "--dec":
|
||
|
akey::dec($this->dest, $key);
|
||
|
break;
|
||
|
case "--set":
|
||
|
akey::set($this->dest, $key, $value);
|
||
|
break;
|
||
|
case "--add":
|
||
|
akey::append($this->dest, $key, $value);
|
||
|
break;
|
||
|
case "--adds":
|
||
|
foreach (A::with($value) as $value) {
|
||
|
akey::append($this->dest, $key, $value);
|
||
|
}
|
||
|
break;
|
||
|
case "--merge":
|
||
|
akey::merge($this->dest, $key, $value);
|
||
|
break;
|
||
|
case "--merges":
|
||
|
foreach (A::with($value) as $value) {
|
||
|
akey::merge($this->dest, $key, $value);
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
throw $this->invalidDef("$action: invalid action");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function updateDest(string $action, string $name, $value): void {
|
||
|
switch ($action) {
|
||
|
case "--inc":
|
||
|
valx::inc($this->dest, $name);
|
||
|
break;
|
||
|
case "--dec":
|
||
|
valx::dec($this->dest, $name);
|
||
|
break;
|
||
|
case "--set":
|
||
|
valx::set($this->dest, $name, $value);
|
||
|
break;
|
||
|
case "--add":
|
||
|
valx::append($this->dest, $name, $value);
|
||
|
break;
|
||
|
case "--adds":
|
||
|
foreach (A::with($value) as $value) {
|
||
|
valx::append($this->dest, $name, $value);
|
||
|
}
|
||
|
break;
|
||
|
case "--merge":
|
||
|
valx::merge($this->dest, $name, $value);
|
||
|
break;
|
||
|
case "--merges":
|
||
|
foreach (A::with($value) as $value) {
|
||
|
valx::merge($this->dest, $name, $value);
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
throw $this->invalidDef("$action: invalid action");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function updateMeta(string $name, $value): void {
|
||
|
$meta = $this->meta;
|
||
|
switch ($name) {
|
||
|
case "args":
|
||
|
$name = $meta["argsname"];
|
||
|
$property = $meta["argsproperty"];
|
||
|
$key = $meta["argskey"];
|
||
|
$action = "--merge";
|
||
|
break;
|
||
|
case "command":
|
||
|
$name = $meta["commandname"];
|
||
|
$property = $meta["commandproperty"];
|
||
|
$key = $meta["commandkey"];
|
||
|
$action = $value !== null? "--add": "--set";
|
||
|
break;
|
||
|
default:
|
||
|
throw IllegalAccessException::unexpected_state();
|
||
|
}
|
||
|
if ($property !== null) {
|
||
|
$this->updateObject($action, $property, $value);
|
||
|
} elseif ($key !== null) {
|
||
|
$this->updateArray($action, $key, $value);
|
||
|
} elseif ($name !== null) {
|
||
|
$this->updateDest($action, $name, $value);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private function doAction($value, ?string $arg, array $def) {
|
||
|
$action = $def["action"];
|
||
|
$name = $def["name"];
|
||
|
$property = $def["property"];
|
||
|
$key = $def["key"];
|
||
|
$ensure_array = $def["ensure_array"];
|
||
|
if ($ensure_array) {
|
||
|
$value = A::with($value);
|
||
|
} elseif (is_array($value)) {
|
||
|
$count = count($value);
|
||
|
if ($count == 0) $value = null;
|
||
|
elseif ($count == 1) $value = $value[0];
|
||
|
}
|
||
|
|
||
|
if ($action === "--meta") {
|
||
|
$this->updateMeta($name, $value);
|
||
|
} else {
|
||
|
if ($action === null) {
|
||
|
# NOP
|
||
|
} elseif (func::is_method($action)) {
|
||
|
# méthode
|
||
|
$func = $action;
|
||
|
$func_args = [$value, $name, $arg, $this->dest, $def];
|
||
|
func::fix_method($func, $this->dest);
|
||
|
func::fix_args($func, $func_args);
|
||
|
func::call($func, ...$func_args);
|
||
|
return;
|
||
|
} elseif (!is_string($action) || substr($action, 0, 2) != "--") {
|
||
|
# fonction statique
|
||
|
$func = $action;
|
||
|
$func_args = [$value, $name, $arg, $this->dest, $def];
|
||
|
func::fix_static($func, $this->dest);
|
||
|
func::fix_args($func, $func_args);
|
||
|
func::call($func, ...$func_args);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if ($property !== null) {
|
||
|
$this->updateObject($action, $property, $value);
|
||
|
} elseif ($key !== null) {
|
||
|
$this->updateArray($action, $key, $value);
|
||
|
} elseif ($name !== null) {
|
||
|
$this->updateDest($action, $name, $value);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||
|
|
||
|
/**
|
||
|
* afficher l'aide correspondant à la liste normalisée de définitions $defs
|
||
|
* et les méta-données $meta
|
||
|
*
|
||
|
* @param bool $showAll faut-il afficher les sections cachées?
|
||
|
*/
|
||
|
function printHelp(array $meta, array $sections, bool $showAll=false) {
|
||
|
if ($meta["prefix"]) echo $meta["prefix"];
|
||
|
|
||
|
if ($meta["purpose"]) {
|
||
|
echo "$meta[name]: $meta[purpose]\n";
|
||
|
} elseif (!$meta["prefix"]) {
|
||
|
echo "$meta[name]\n";
|
||
|
}
|
||
|
|
||
|
if ($meta["usage"]) {
|
||
|
echo "\nUSAGE\n";
|
||
|
foreach (A::with($meta["usage"]) as $usage) {
|
||
|
echo " $meta[name] $usage\n";
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($meta["description"]) {
|
||
|
echo "\n$meta[description]\n";
|
||
|
}
|
||
|
|
||
|
$firstSection = null; # la première section est la section par défaut
|
||
|
# mais il faut l'afficher en dernier...
|
||
|
foreach ($sections as $section) {
|
||
|
if (!$section["show"] && !$showAll) continue;
|
||
|
|
||
|
if ($firstSection === null) {
|
||
|
$firstSection = $section;
|
||
|
} else {
|
||
|
# dans les autres sections, afficher tel quel
|
||
|
echo "\n";
|
||
|
if ($section["title"]) echo "$section[title]\n";
|
||
|
if ($section["prefix"]) echo "$section[prefix]\n";
|
||
|
foreach ($section["defs"] as $def) {
|
||
|
$this->printDef($def);
|
||
|
}
|
||
|
if ($section["suffix"]) echo "$section[suffix]\n";
|
||
|
}
|
||
|
}
|
||
|
if ($firstSection !== null) {
|
||
|
$section = $firstSection;
|
||
|
# dans la section par défaut, séparer commandes et options
|
||
|
$options = [];
|
||
|
$commands = [];
|
||
|
foreach ($section["defs"] as $def) {
|
||
|
if ($this->isGroup($def)) {
|
||
|
if ($def[1]["is_option"]) $options[] = $def;
|
||
|
elseif ($def[1]["is_command"]) $commands[] = $def;
|
||
|
} elseif ($def["is_option"]) {
|
||
|
$options[] = $def;
|
||
|
} elseif ($def["is_command"]) {
|
||
|
$commands[] = $def;
|
||
|
}
|
||
|
}
|
||
|
if ($options) {
|
||
|
echo "\nOPTIONS\n";
|
||
|
foreach ($options as $def) {
|
||
|
$this->printDef($def);
|
||
|
}
|
||
|
}
|
||
|
/** @var IDynamicCommand $dc */
|
||
|
$dc = $meta["dynamic_command"];
|
||
|
$dcCommands = $dc !== null? $dc->getCommands(): null;
|
||
|
if ($commands || $dcCommands) echo "\nCOMMANDES\n";
|
||
|
if ($commands) {
|
||
|
foreach ($commands as $def) {
|
||
|
$this->printDef($def);
|
||
|
}
|
||
|
}
|
||
|
if ($dcCommands) {
|
||
|
foreach ($dcCommands as $command) {
|
||
|
$dyncdefs = [];
|
||
|
$kind = $this->checkDyncdefs($dc, $command, $dyncdefs, true);
|
||
|
if ($kind == "cmd") {
|
||
|
$this->printDef($dyncdefs[$command]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ($meta["suffix"]) echo $meta["suffix"];
|
||
|
}
|
||
|
|
||
|
private function printDef(array $def): void {
|
||
|
if ($this->isGroup($def)) {
|
||
|
$count = count($def);
|
||
|
$help = null;
|
||
|
$first = true;
|
||
|
for ($i = 1; $i < $count; $i++) {
|
||
|
$defi = $def[$i];
|
||
|
$seq = A::split_assoc($defi)[0];
|
||
|
echo " ".implode(", ", $seq);
|
||
|
if ($defi["have_args"]) echo " ".$defi["argsdesc"];
|
||
|
echo "\n";
|
||
|
|
||
|
if ($first) $help = $defi["help"];
|
||
|
$first = false;
|
||
|
}
|
||
|
} else {
|
||
|
$seq = A::split_assoc($def)[0];
|
||
|
echo " ".implode(", ", $seq);
|
||
|
if ($def["have_args"]) echo " ".$def["argsdesc"];
|
||
|
echo "\n";
|
||
|
$help = $def["help"];
|
||
|
}
|
||
|
if ($help) {
|
||
|
$help = str_replace("\n", "\n ", $help);
|
||
|
echo " $help\n";
|
||
|
}
|
||
|
}
|
||
|
}
|