Файловый менеджер - Редактировать - /home/clickysoft/public_html/charliapp-v2.clickysoft.net/vendor/zircote/swagger-php/src/Generator.php
Назад
<?php declare(strict_types=1); /** * @license Apache 2.0 */ namespace OpenApi; use Doctrine\Common\Annotations\AnnotationRegistry; use OpenApi\Analysers\AnalyserInterface; use OpenApi\Analysers\AttributeAnnotationFactory; use OpenApi\Analysers\DocBlockAnnotationFactory; use OpenApi\Analysers\ReflectionAnalyser; use OpenApi\Annotations as OA; use OpenApi\Loggers\DefaultLogger; use OpenApi\Processors\ProcessorInterface; use Psr\Log\LoggerInterface; /** * OpenApi spec generator. * * Scans PHP source code and generates OpenApi specifications from the found OpenApi annotations. * * This is an object-oriented alternative to using the now deprecated `\OpenApi\scan()` function and * static class properties of the `Analyzer` and `Analysis` classes. */ class Generator { /** * Allows Annotation classes to know the context of the annotation that is being processed. * * @var Context|null */ public static $context; /** @var string Magic value to differentiate between null and undefined. */ public const UNDEFINED = '@OA\Generator::UNDEFINED🙈'; /** @var array<string,string> */ public const DEFAULT_ALIASES = ['oa' => 'OpenApi\\Annotations']; /** @var array<string> */ public const DEFAULT_NAMESPACES = ['OpenApi\\Annotations\\']; /** @var array<string,string> Map of namespace aliases to be supported by doctrine. */ protected $aliases; /** @var array<string>|null List of annotation namespaces to be autoloaded by doctrine. */ protected $namespaces; /** @var AnalyserInterface|null The configured analyzer. */ protected $analyser = null; /** @var array<string,mixed> */ protected $config = []; /** @var Pipeline|null */ protected $processorPipeline = null; /** @var LoggerInterface|null PSR logger. */ protected $logger = null; /** * OpenApi version override. * * If set, it will override the version set in the `OpenApi` annotation. * * Due to the order of processing any conditional code using this (via `Context::$version`) * must come only after the analysis is finished. * * @var string|null */ protected $version = null; private $configStack; public function __construct(?LoggerInterface $logger = null) { $this->logger = $logger; $this->setAliases(self::DEFAULT_ALIASES); $this->setNamespaces(self::DEFAULT_NAMESPACES); // kinda config stack to stay BC... // @deprecated Can be removed once doctrine/annotations 2.0 becomes mandatory $this->configStack = new class() { protected $generator; public function push(Generator $generator): void { $this->generator = $generator; /* @phpstan-ignore-next-line */ if (class_exists(AnnotationRegistry::class, true) && method_exists(AnnotationRegistry::class, 'registerLoader')) { // keeping track of &this->generator allows to 'disable' the loader after we are done; // no unload, unfortunately :/ $gref = &$this->generator; AnnotationRegistry::registerLoader( function (string $class) use (&$gref): bool { if ($gref) { foreach ($gref->getNamespaces() as $namespace) { if (strtolower(substr($class, 0, strlen($namespace))) === strtolower($namespace)) { $loaded = class_exists($class); if (!$loaded && $namespace === 'OpenApi\\Annotations\\') { if (in_array(strtolower(substr($class, 20)), ['definition', 'path'])) { // Detected an 2.x annotation? throw new OpenApiException('The annotation @SWG\\' . substr($class, 20) . '() is deprecated. Found in ' . Generator::$context . "\nFor more information read the migration guide: https://github.com/zircote/swagger-php/blob/master/docs/Migrating-to-v3.md"); } } return $loaded; } } } return false; } ); } } public function pop(): void { $this->generator = null; } }; } public static function isDefault($value): bool { return $value === Generator::UNDEFINED; } /** * @return array<string> */ public function getAliases(): array { return $this->aliases; } public function addAlias(string $alias, string $namespace): Generator { $this->aliases[$alias] = $namespace; return $this; } public function setAliases(array $aliases): Generator { $this->aliases = $aliases; return $this; } /** * @return array<string>|null */ public function getNamespaces(): ?array { return $this->namespaces; } public function addNamespace(string $namespace): Generator { $namespaces = (array) $this->getNamespaces(); $namespaces[] = $namespace; return $this->setNamespaces(array_unique($namespaces)); } public function setNamespaces(?array $namespaces): Generator { $this->namespaces = $namespaces; return $this; } public function getAnalyser(): AnalyserInterface { $this->analyser = $this->analyser ?: new ReflectionAnalyser([new DocBlockAnnotationFactory(), new AttributeAnnotationFactory()]); $this->analyser->setGenerator($this); return $this->analyser; } public function setAnalyser(?AnalyserInterface $analyser): Generator { $this->analyser = $analyser; return $this; } public function getDefaultConfig(): array { return [ 'operationId' => [ 'hash' => true, ], ]; } public function getConfig(): array { return $this->config + $this->getDefaultConfig(); } protected function normaliseConfig(array $config): array { $normalised = []; foreach ($config as $key => $value) { if (is_numeric($key)) { $token = explode('=', $value); if (2 == count($token)) { // 'operationId.hash=false' [$key, $value] = $token; } } if (in_array($value, ['true', 'false'])) { $value = 'true' == $value; } if ($isList = ('[]' == substr($key, -2))) { $key = substr($key, 0, -2); } $token = explode('.', $key); if (2 == count($token)) { // 'operationId.hash' => false // namespaced / processor if ($isList) { $normalised[$token[0]][$token[1]][] = $value; } else { $normalised[$token[0]][$token[1]] = $value; } } else { if ($isList) { $normalised[$key][] = $value; } else { $normalised[$key] = $value; } } } return $normalised; } /** * Set generator and/or processor config. * * @param array<string,mixed> $config */ public function setConfig(array $config): Generator { $this->config = $this->normaliseConfig($config) + $this->config; return $this; } public function getProcessorPipeline(): Pipeline { if (null === $this->processorPipeline) { $this->processorPipeline = new Pipeline([ new Processors\DocBlockDescriptions(), new Processors\MergeIntoOpenApi(), new Processors\MergeIntoComponents(), new Processors\ExpandClasses(), new Processors\ExpandInterfaces(), new Processors\ExpandTraits(), new Processors\ExpandEnums(), new Processors\AugmentSchemas(), new Processors\AugmentRequestBody(), new Processors\AugmentProperties(), new Processors\BuildPaths(), new Processors\AugmentParameters(), new Processors\AugmentRefs(), new Processors\MergeJsonContent(), new Processors\MergeXmlContent(), new Processors\OperationId(), new Processors\CleanUnmerged(), new Processors\PathFilter(), new Processors\CleanUnusedComponents(), new Processors\AugmentTags(), ]); } $config = $this->getConfig(); $walker = function (callable $pipe) use ($config) { $rc = new \ReflectionClass($pipe); // apply config $processorKey = lcfirst($rc->getShortName()); if (array_key_exists($processorKey, $config)) { foreach ($config[$processorKey] as $name => $value) { $setter = 'set' . ucfirst($name); if (method_exists($pipe, $setter)) { $pipe->{$setter}($value); } } } }; return $this->processorPipeline->walk($walker); } public function setProcessorPipeline(?Pipeline $processor): Generator { $this->processorPipeline = $processor; return $this; } /** * Chainable method that allows to modify the processor pipeline. * * @param callable $with callable with the current processor pipeline passed in */ public function withProcessor(callable $with): Generator { $with($this->getProcessorPipeline()); return $this; } /** * @return array<ProcessorInterface|callable> * * @deprecated */ public function getProcessors(): array { return $this->getProcessorPipeline()->pipes(); } /** * @param array<ProcessorInterface|callable>|null $processors * * @deprecated */ public function setProcessors(?array $processors): Generator { $this->processorPipeline = null !== $processors ? new Pipeline($processors) : null; return $this; } /** * @param callable|ProcessorInterface $processor * @param class-string|null $before * * @deprecated */ public function addProcessor($processor, ?string $before = null): Generator { $processors = $this->processorPipeline ?: $this->getProcessorPipeline(); if (!$before) { $processors->add($processor); } else { $processors->insert($processor, $before); } $this->processorPipeline = $processors; return $this; } /** * @param callable|ProcessorInterface $processor * * @deprecated */ public function removeProcessor($processor, bool $silent = false): Generator { $processors = $this->processorPipeline ?: $this->getProcessorPipeline(); $processors->remove($processor); $this->processorPipeline = $processors; return $this; } /** * Update/replace an existing processor with a new one. * * @param ProcessorInterface|callable $processor the new processor * @param null|callable $matcher Optional matcher callable to identify the processor to replace. * If none given, matching is based on the processors class. * * @deprecated */ public function updateProcessor($processor, ?callable $matcher = null): Generator { $matcher = $matcher ?: function ($other) use ($processor): bool { $otherClass = get_class($other); return $processor instanceof $otherClass; }; $processors = array_map(function ($other) use ($processor, $matcher) { return $matcher($other) ? $processor : $other; }, $this->getProcessors()); $this->setProcessors($processors); return $this; } public function getLogger(): ?LoggerInterface { return $this->logger ?: new DefaultLogger(); } public function getVersion(): ?string { return $this->version; } public function setVersion(?string $version): Generator { $this->version = $version; return $this; } public static function scan(iterable $sources, array $options = []): ?OA\OpenApi { // merge with defaults $config = $options + [ 'aliases' => self::DEFAULT_ALIASES, 'namespaces' => self::DEFAULT_NAMESPACES, 'analyser' => null, 'analysis' => null, 'processor' => null, 'processors' => null, 'config' => [], 'logger' => null, 'validate' => true, 'version' => null, ]; $processorPipeline = $config['processor'] ?? ($config['processors'] ? new Pipeline($config['processors']) : null); return (new Generator($config['logger'])) ->setVersion($config['version']) ->setAliases($config['aliases']) ->setNamespaces($config['namespaces']) ->setAnalyser($config['analyser']) ->setProcessorPipeline($processorPipeline) ->setConfig($config['config']) ->generate($sources, $config['analysis'], $config['validate']); } /** * Run code in the context of this generator. * * @param callable $callable Callable in the form of * `function(Generator $generator, Analysis $analysis, Context $context): mixed` * * @return mixed the result of the `callable` */ public function withContext(callable $callable) { $rootContext = new Context([ 'version' => $this->getVersion(), 'logger' => $this->getLogger(), ]); $analysis = new Analysis([], $rootContext); $this->configStack->push($this); try { return $callable($this, $analysis, $rootContext); } finally { $this->configStack->pop(); } } /** * Generate OpenAPI spec by scanning the given source files. * * @param iterable $sources PHP source files to scan. * Supported sources: * * string - file / directory name * * \SplFileInfo * * \Symfony\Component\Finder\Finder * @param null|Analysis $analysis custom analysis instance * @param bool $validate flag to enable/disable validation of the returned spec */ public function generate(iterable $sources, ?Analysis $analysis = null, bool $validate = true): ?OA\OpenApi { $rootContext = new Context([ 'version' => $this->getVersion(), 'logger' => $this->getLogger(), ]); $analysis = $analysis ?: new Analysis([], $rootContext); $analysis->context = $analysis->context ?: $rootContext; $this->configStack->push($this); try { $this->scanSources($sources, $analysis, $rootContext); // post-processing $this->getProcessorPipeline()->process($analysis); if ($analysis->openapi) { // overwrite default/annotated version $analysis->openapi->openapi = $this->getVersion() ?: $analysis->openapi->openapi; // update context to provide the same to validation/serialisation code $rootContext->version = $analysis->openapi->openapi; } // validation if ($validate) { $analysis->validate(); } } finally { $this->configStack->pop(); } return $analysis->openapi; } protected function scanSources(iterable $sources, Analysis $analysis, Context $rootContext): void { $analyser = $this->getAnalyser(); foreach ($sources as $source) { if (is_iterable($source)) { $this->scanSources($source, $analysis, $rootContext); } else { $resolvedSource = $source instanceof \SplFileInfo ? $source->getPathname() : realpath($source); if (!$resolvedSource) { $rootContext->logger->warning(sprintf('Skipping invalid source: %s', $source)); continue; } if (is_dir($resolvedSource)) { $this->scanSources(Util::finder($resolvedSource), $analysis, $rootContext); } else { $rootContext->logger->debug(sprintf('Analysing source: %s', $resolvedSource)); $analysis->addAnalysis($analyser->fromFile($resolvedSource, $rootContext)); } } } } }
| ver. 1.4 |
Github
|
.
| PHP 8.1.29 | Генерация страницы: 0.01 |
proxy
|
phpinfo
|
Настройка