// +---------------------------------------------------------------------- namespace think\console\input; class Definition { /** * @var Argument[] */ private $arguments; private $requiredCount; private $hasAnArrayArgument = false; private $hasOptional; /** * @var Option[] */ private $options; private $shortcuts; /** * 构造方法 * @param array $definition * @api */ public function __construct(array $definition = []) { $this->setDefinition($definition); } /** * 设置指令的定义 * @param array $definition 定义的数组 */ public function setDefinition(array $definition) { $arguments = []; $options = []; foreach ($definition as $item) { if ($item instanceof Option) { $options[] = $item; } else { $arguments[] = $item; } } $this->setArguments($arguments); $this->setOptions($options); } /** * 设置参数 * @param Argument[] $arguments 参数数组 */ public function setArguments($arguments = []) { $this->arguments = []; $this->requiredCount = 0; $this->hasOptional = false; $this->hasAnArrayArgument = false; $this->addArguments($arguments); } /** * 添加参数 * @param Argument[] $arguments 参数数组 * @api */ public function addArguments($arguments = []) { if (null !== $arguments) { foreach ($arguments as $argument) { $this->addArgument($argument); } } } /** * 添加一个参数 * @param Argument $argument 参数 * @throws \LogicException */ public function addArgument(Argument $argument) { if (isset($this->arguments[$argument->getName()])) { throw new \LogicException(sprintf('An argument with name "%s" already exists.', $argument->getName())); } if ($this->hasAnArrayArgument) { throw new \LogicException('Cannot add an argument after an array argument.'); } if ($argument->isRequired() && $this->hasOptional) { throw new \LogicException('Cannot add a required argument after an optional one.'); } if ($argument->isArray()) { $this->hasAnArrayArgument = true; } if ($argument->isRequired()) { ++$this->requiredCount; } else { $this->hasOptional = true; } $this->arguments[$argument->getName()] = $argument; } /** * 根据名称或者位置获取参数 * @param string|int $name 参数名或者位置 * @return Argument 参数 * @throws \InvalidArgumentException */ public function getArgument($name) { if (!$this->hasArgument($name)) { throw new \InvalidArgumentException(sprintf('The "%s" argument does not exist.', $name)); } $arguments = is_int($name) ? array_values($this->arguments) : $this->arguments; return $arguments[$name]; } /** * 根据名称或位置检查是否具有某个参数 * @param string|int $name 参数名或者位置 * @return bool * @api */ public function hasArgument($name) { $arguments = is_int($name) ? array_values($this->arguments) : $this->arguments; return isset($arguments[$name]); } /** * 获取所有的参数 * @return Argument[] 参数数组 */ public function getArguments() { return $this->arguments; } /** * 获取参数数量 * @return int */ public function getArgumentCount() { return $this->hasAnArrayArgument ? PHP_INT_MAX : count($this->arguments); } /** * 获取必填的参数的数量 * @return int */ public function getArgumentRequiredCount() { return $this->requiredCount; } /** * 获取参数默认值 * @return array */ public function getArgumentDefaults() { $values = []; foreach ($this->arguments as $argument) { $values[$argument->getName()] = $argument->getDefault(); } return $values; } /** * 设置选项 * @param Option[] $options 选项数组 */ public function setOptions($options = []) { $this->options = []; $this->shortcuts = []; $this->addOptions($options); } /** * 添加选项 * @param Option[] $options 选项数组 * @api */ public function addOptions($options = []) { foreach ($options as $option) { $this->addOption($option); } } /** * 添加一个选项 * @param Option $option 选项 * @throws \LogicException * @api */ public function addOption(Option $option) { if (isset($this->options[$option->getName()]) && !$option->equals($this->options[$option->getName()])) { throw new \LogicException(sprintf('An option named "%s" already exists.', $option->getName())); } if ($option->getShortcut()) { foreach (explode('|', $option->getShortcut()) as $shortcut) { if (isset($this->shortcuts[$shortcut]) && !$option->equals($this->options[$this->shortcuts[$shortcut]]) ) { throw new \LogicException(sprintf('An option with shortcut "%s" already exists.', $shortcut)); } } } $this->options[$option->getName()] = $option; if ($option->getShortcut()) { foreach (explode('|', $option->getShortcut()) as $shortcut) { $this->shortcuts[$shortcut] = $option->getName(); } } } /** * 根据名称获取选项 * @param string $name 选项名 * @return Option * @throws \InvalidArgumentException * @api */ public function getOption($name) { if (!$this->hasOption($name)) { throw new \InvalidArgumentException(sprintf('The "--%s" option does not exist.', $name)); } return $this->options[$name]; } /** * 根据名称检查是否有这个选项 * @param string $name 选项名 * @return bool * @api */ public function hasOption($name) { return isset($this->options[$name]); } /** * 获取所有选项 * @return Option[] * @api */ public function getOptions() { return $this->options; } /** * 根据名称检查某个选项是否有短名称 * @param string $name 短名称 * @return bool */ public function hasShortcut($name) { return isset($this->shortcuts[$name]); } /** * 根据短名称获取选项 * @param string $shortcut 短名称 * @return Option */ public function getOptionForShortcut($shortcut) { return $this->getOption($this->shortcutToName($shortcut)); } /** * 获取所有选项的默认值 * @return array */ public function getOptionDefaults() { $values = []; foreach ($this->options as $option) { $values[$option->getName()] = $option->getDefault(); } return $values; } /** * 根据短名称获取选项名 * @param string $shortcut 短名称 * @return string * @throws \InvalidArgumentException */ private function shortcutToName($shortcut) { if (!isset($this->shortcuts[$shortcut])) { throw new \InvalidArgumentException(sprintf('The "-%s" option does not exist.', $shortcut)); } return $this->shortcuts[$shortcut]; } /** * 获取该指令的介绍 * @param bool $short 是否简洁介绍 * @return string */ public function getSynopsis($short = false) { $elements = []; if ($short && $this->getOptions()) { $elements[] = '[options]'; } elseif (!$short) { foreach ($this->getOptions() as $option) { $value = ''; if ($option->acceptValue()) { $value = sprintf(' %s%s%s', $option->isValueOptional() ? '[' : '', strtoupper($option->getName()), $option->isValueOptional() ? ']' : ''); } $shortcut = $option->getShortcut() ? sprintf('-%s|', $option->getShortcut()) : ''; $elements[] = sprintf('[%s--%s%s]', $shortcut, $option->getName(), $value); } } if (count($elements) && $this->getArguments()) { $elements[] = '[--]'; } foreach ($this->getArguments() as $argument) { $element = '<' . $argument->getName() . '>'; if (!$argument->isRequired()) { $element = '[' . $element . ']'; } elseif ($argument->isArray()) { $element .= ' (' . $element . ')'; } if ($argument->isArray()) { $element .= '...'; } $elements[] = $element; } return implode(' ', $elements); } }