urlShortener = $urlShortener; $this->domainConfig = $domainConfig; } protected function configure(): void { $this ->setName(self::NAME) ->setAliases(self::ALIASES) ->setDescription('Generates a short URL for provided long URL and returns it') ->addArgument('longUrl', InputArgument::REQUIRED, 'The long URL to parse') ->addOption( 'tags', 't', InputOption::VALUE_IS_ARRAY | InputOption::VALUE_REQUIRED, 'Tags to apply to the new short URL' ) ->addOption( 'validSince', 's', InputOption::VALUE_REQUIRED, 'The date from which this short URL will be valid. ' . 'If someone tries to access it before this date, it will not be found.' ) ->addOption( 'validUntil', 'u', InputOption::VALUE_REQUIRED, 'The date until which this short URL will be valid. ' . 'If someone tries to access it after this date, it will not be found.' ) ->addOption( 'customSlug', 'c', InputOption::VALUE_REQUIRED, 'If provided, this slug will be used instead of generating a short code' ) ->addOption( 'maxVisits', 'm', InputOption::VALUE_REQUIRED, 'This will limit the number of visits for this short URL.' ) ->addOption( 'findIfExists', 'f', InputOption::VALUE_NONE, 'This will force existing matching URL to be returned if found, instead of creating a new one.' ); } protected function interact(InputInterface $input, OutputInterface $output): void { $io = new SymfonyStyle($input, $output); $longUrl = $input->getArgument('longUrl'); if (! empty($longUrl)) { return; } $longUrl = $io->ask('A long URL was not provided. Which URL do you want to be shortened?'); if (! empty($longUrl)) { $input->setArgument('longUrl', $longUrl); } } protected function execute(InputInterface $input, OutputInterface $output): ?int { $io = new SymfonyStyle($input, $output); $longUrl = $input->getArgument('longUrl'); if (empty($longUrl)) { $io->error('A URL was not provided!'); return ExitCodes::EXIT_FAILURE; } $explodeWithComma = curry('explode')(','); $tags = unique(flatten(array_map($explodeWithComma, $input->getOption('tags')))); $customSlug = $input->getOption('customSlug'); $maxVisits = $input->getOption('maxVisits'); try { $shortCode = $this->urlShortener->urlToShortCode( new Uri($longUrl), $tags, ShortUrlMeta::createFromParams( $this->getOptionalDate($input, 'validSince'), $this->getOptionalDate($input, 'validUntil'), $customSlug, $maxVisits !== null ? (int) $maxVisits : null, $input->getOption('findIfExists') ) )->getShortCode(); $shortUrl = $this->buildShortUrl($this->domainConfig, $shortCode); $io->writeln([ sprintf('Processed long URL: %s', $longUrl), sprintf('Generated short URL: %s', $shortUrl), ]); return ExitCodes::EXIT_SUCCESS; } catch (InvalidUrlException $e) { $io->error(sprintf('Provided URL "%s" is invalid. Try with a different one.', $longUrl)); return ExitCodes::EXIT_FAILURE; } catch (NonUniqueSlugException $e) { $io->error( sprintf('Provided slug "%s" is already in use by another URL. Try with a different one.', $customSlug) ); return ExitCodes::EXIT_FAILURE; } } private function getOptionalDate(InputInterface $input, string $fieldName): ?Chronos { $since = $input->getOption($fieldName); return $since !== null ? Chronos::parse($since) : null; } }