209 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			209 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			PHP
		
	
	
<?php
 | 
						|
 | 
						|
/*
 | 
						|
 * This file is part of the Symfony package.
 | 
						|
 *
 | 
						|
 * (c) Fabien Potencier <fabien@symfony.com>
 | 
						|
 *
 | 
						|
 * For the full copyright and license information, please view the LICENSE
 | 
						|
 * file that was distributed with this source code.
 | 
						|
 */
 | 
						|
 | 
						|
namespace Symfony\Component\Uid\Command;
 | 
						|
 | 
						|
use Symfony\Component\Console\Attribute\AsCommand;
 | 
						|
use Symfony\Component\Console\Command\Command;
 | 
						|
use Symfony\Component\Console\Completion\CompletionInput;
 | 
						|
use Symfony\Component\Console\Completion\CompletionSuggestions;
 | 
						|
use Symfony\Component\Console\Input\InputInterface;
 | 
						|
use Symfony\Component\Console\Input\InputOption;
 | 
						|
use Symfony\Component\Console\Output\ConsoleOutputInterface;
 | 
						|
use Symfony\Component\Console\Output\OutputInterface;
 | 
						|
use Symfony\Component\Console\Style\SymfonyStyle;
 | 
						|
use Symfony\Component\Uid\Factory\UuidFactory;
 | 
						|
use Symfony\Component\Uid\Uuid;
 | 
						|
 | 
						|
#[AsCommand(name: 'uuid:generate', description: 'Generate a UUID')]
 | 
						|
class GenerateUuidCommand extends Command
 | 
						|
{
 | 
						|
    private UuidFactory $factory;
 | 
						|
 | 
						|
    public function __construct(?UuidFactory $factory = null)
 | 
						|
    {
 | 
						|
        $this->factory = $factory ?? new UuidFactory();
 | 
						|
 | 
						|
        parent::__construct();
 | 
						|
    }
 | 
						|
 | 
						|
    protected function configure(): void
 | 
						|
    {
 | 
						|
        $this
 | 
						|
            ->setDefinition([
 | 
						|
                new InputOption('time-based', null, InputOption::VALUE_REQUIRED, 'The timestamp, to generate a time-based UUID: a parsable date/time string'),
 | 
						|
                new InputOption('node', null, InputOption::VALUE_REQUIRED, 'The UUID whose node part should be used as the node of the generated UUID'),
 | 
						|
                new InputOption('name-based', null, InputOption::VALUE_REQUIRED, 'The name, to generate a name-based UUID'),
 | 
						|
                new InputOption('namespace', null, InputOption::VALUE_REQUIRED, 'The UUID to use at the namespace for named-based UUIDs, predefined namespaces keywords "dns", "url", "oid" and "x500" are accepted'),
 | 
						|
                new InputOption('random-based', null, InputOption::VALUE_NONE, 'To generate a random-based UUID'),
 | 
						|
                new InputOption('count', 'c', InputOption::VALUE_REQUIRED, 'The number of UUID to generate', 1),
 | 
						|
                new InputOption('format', 'f', InputOption::VALUE_REQUIRED, sprintf('The UUID output format ("%s")', implode('", "', $this->getAvailableFormatOptions())), 'rfc4122'),
 | 
						|
            ])
 | 
						|
            ->setHelp(<<<'EOF'
 | 
						|
The <info>%command.name%</info> generates a UUID.
 | 
						|
 | 
						|
    <info>php %command.full_name%</info>
 | 
						|
 | 
						|
To generate a time-based UUID:
 | 
						|
 | 
						|
    <info>php %command.full_name% --time-based=now</info>
 | 
						|
 | 
						|
To specify a time-based UUID's node:
 | 
						|
 | 
						|
    <info>php %command.full_name% --time-based=@1613480254 --node=fb3502dc-137e-4849-8886-ac90d07f64a7</info>
 | 
						|
 | 
						|
To generate a name-based UUID:
 | 
						|
 | 
						|
    <info>php %command.full_name% --name-based=foo</info>
 | 
						|
 | 
						|
To specify a name-based UUID's namespace:
 | 
						|
 | 
						|
    <info>php %command.full_name% --name-based=bar --namespace=fb3502dc-137e-4849-8886-ac90d07f64a7</info>
 | 
						|
 | 
						|
To generate a random-based UUID:
 | 
						|
 | 
						|
    <info>php %command.full_name% --random-based</info>
 | 
						|
 | 
						|
To generate several UUIDs:
 | 
						|
 | 
						|
    <info>php %command.full_name% --count=10</info>
 | 
						|
 | 
						|
To output a specific format:
 | 
						|
 | 
						|
    <info>php %command.full_name% --format=base58</info>
 | 
						|
EOF
 | 
						|
            )
 | 
						|
        ;
 | 
						|
    }
 | 
						|
 | 
						|
    protected function execute(InputInterface $input, OutputInterface $output): int
 | 
						|
    {
 | 
						|
        $io = new SymfonyStyle($input, $output instanceof ConsoleOutputInterface ? $output->getErrorOutput() : $output);
 | 
						|
 | 
						|
        $time = $input->getOption('time-based');
 | 
						|
        $node = $input->getOption('node');
 | 
						|
        $name = $input->getOption('name-based');
 | 
						|
        $namespace = $input->getOption('namespace');
 | 
						|
        $random = $input->getOption('random-based');
 | 
						|
 | 
						|
        if (false !== ($time ?? $name ?? $random) && 1 < ((null !== $time) + (null !== $name) + $random)) {
 | 
						|
            $io->error('Only one of "--time-based", "--name-based" or "--random-based" can be provided at a time.');
 | 
						|
 | 
						|
            return 1;
 | 
						|
        }
 | 
						|
 | 
						|
        if (null === $time && null !== $node) {
 | 
						|
            $io->error('Option "--node" can only be used with "--time-based".');
 | 
						|
 | 
						|
            return 1;
 | 
						|
        }
 | 
						|
 | 
						|
        if (null === $name && null !== $namespace) {
 | 
						|
            $io->error('Option "--namespace" can only be used with "--name-based".');
 | 
						|
 | 
						|
            return 1;
 | 
						|
        }
 | 
						|
 | 
						|
        switch (true) {
 | 
						|
            case null !== $time:
 | 
						|
                if (null !== $node) {
 | 
						|
                    try {
 | 
						|
                        $node = Uuid::fromString($node);
 | 
						|
                    } catch (\InvalidArgumentException $e) {
 | 
						|
                        $io->error(sprintf('Invalid node "%s": %s', $node, $e->getMessage()));
 | 
						|
 | 
						|
                        return 1;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                try {
 | 
						|
                    new \DateTimeImmutable($time);
 | 
						|
                } catch (\Exception $e) {
 | 
						|
                    $io->error(sprintf('Invalid timestamp "%s": %s', $time, str_replace('DateTimeImmutable::__construct(): ', '', $e->getMessage())));
 | 
						|
 | 
						|
                    return 1;
 | 
						|
                }
 | 
						|
 | 
						|
                $create = fn (): Uuid => $this->factory->timeBased($node)->create(new \DateTimeImmutable($time));
 | 
						|
                break;
 | 
						|
 | 
						|
            case null !== $name:
 | 
						|
                if ($namespace && !\in_array($namespace, ['dns', 'url', 'oid', 'x500'], true)) {
 | 
						|
                    try {
 | 
						|
                        $namespace = Uuid::fromString($namespace);
 | 
						|
                    } catch (\InvalidArgumentException $e) {
 | 
						|
                        $io->error(sprintf('Invalid namespace "%s": %s', $namespace, $e->getMessage()));
 | 
						|
 | 
						|
                        return 1;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                $create = function () use ($namespace, $name): Uuid {
 | 
						|
                    try {
 | 
						|
                        $factory = $this->factory->nameBased($namespace);
 | 
						|
                    } catch (\LogicException) {
 | 
						|
                        throw new \InvalidArgumentException('Missing namespace: use the "--namespace" option or configure a default namespace in the underlying factory.');
 | 
						|
                    }
 | 
						|
 | 
						|
                    return $factory->create($name);
 | 
						|
                };
 | 
						|
                break;
 | 
						|
 | 
						|
            case $random:
 | 
						|
                $create = $this->factory->randomBased()->create(...);
 | 
						|
                break;
 | 
						|
 | 
						|
            default:
 | 
						|
                $create = $this->factory->create(...);
 | 
						|
                break;
 | 
						|
        }
 | 
						|
 | 
						|
        $formatOption = $input->getOption('format');
 | 
						|
 | 
						|
        if (\in_array($formatOption, $this->getAvailableFormatOptions())) {
 | 
						|
            $format = 'to'.ucfirst($formatOption);
 | 
						|
        } else {
 | 
						|
            $io->error(sprintf('Invalid format "%s", supported formats are "%s".', $formatOption, implode('", "', $this->getAvailableFormatOptions())));
 | 
						|
 | 
						|
            return 1;
 | 
						|
        }
 | 
						|
 | 
						|
        $count = (int) $input->getOption('count');
 | 
						|
        try {
 | 
						|
            for ($i = 0; $i < $count; ++$i) {
 | 
						|
                $output->writeln($create()->$format());
 | 
						|
            }
 | 
						|
        } catch (\Exception $e) {
 | 
						|
            $io->error($e->getMessage());
 | 
						|
 | 
						|
            return 1;
 | 
						|
        }
 | 
						|
 | 
						|
        return 0;
 | 
						|
    }
 | 
						|
 | 
						|
    public function complete(CompletionInput $input, CompletionSuggestions $suggestions): void
 | 
						|
    {
 | 
						|
        if ($input->mustSuggestOptionValuesFor('format')) {
 | 
						|
            $suggestions->suggestValues($this->getAvailableFormatOptions());
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    private function getAvailableFormatOptions(): array
 | 
						|
    {
 | 
						|
        return [
 | 
						|
            'base32',
 | 
						|
            'base58',
 | 
						|
            'rfc4122',
 | 
						|
        ];
 | 
						|
    }
 | 
						|
}
 |