*/ private $properties = []; public function __construct() { $schema = JsonFile::parseJson((string) file_get_contents(__DIR__.'/../../../res/composer-schema.json')); /** * @var string $prop */ foreach ($schema['properties']['config']['properties'] as $prop => $conf) { $type = $this->parseType($conf, $prop); $this->properties[$prop] = $type; } } public function getClass(): string { return Config::class; } public function isMethodSupported(MethodReflection $methodReflection): bool { return strtolower($methodReflection->getName()) === 'get'; } public function getTypeFromMethodCall( MethodReflection $methodReflection, MethodCall $methodCall, Scope $scope, ): Type { $args = $methodCall->getArgs(); $defaultReturn = ParametersAcceptorSelector::selectSingle($methodReflection->getVariants())->getReturnType(); if (count($args) < 1) { return $defaultReturn; } $keyType = $scope->getType($args[0]->value); if (method_exists($keyType, 'getConstantStrings')) { // @phpstan-ignore-line - depending on PHPStan version, this method will always exist, or not. $strings = $keyType->getConstantStrings(); } else { // for compat with old phpstan versions, we use a deprecated phpstan method. $strings = TypeUtils::getConstantStrings($keyType); // @phpstan-ignore-line ignore deprecation } if ($strings !== []) { $types = []; foreach($strings as $string) { if (!isset($this->properties[$string->getValue()])) { return $defaultReturn; } $types[] = $this->properties[$string->getValue()]; } return TypeCombinator::union(...$types); } return $defaultReturn; } /** * @param array $def */ private function parseType(array $def, string $path): Type { if (isset($def['type'])) { $types = []; foreach ((array) $def['type'] as $type) { switch ($type) { case 'integer': if (in_array($path, ['process-timeout', 'cache-ttl', 'cache-files-ttl', 'cache-files-maxsize'], true)) { $types[] = IntegerRangeType::createAllGreaterThanOrEqualTo(0); } else { $types[] = new IntegerType(); } break; case 'string': if ($path === 'cache-files-maxsize') { // passthru, skip as it is always converted to int } elseif ($path === 'discard-changes') { $types[] = new ConstantStringType('stash'); } elseif ($path === 'use-parent-dir') { $types[] = new ConstantStringType('prompt'); } elseif ($path === 'store-auths') { $types[] = new ConstantStringType('prompt'); } elseif ($path === 'platform-check') { $types[] = new ConstantStringType('php-only'); } elseif ($path === 'github-protocols') { $types[] = new UnionType([new ConstantStringType('git'), new ConstantStringType('https'), new ConstantStringType('ssh'), new ConstantStringType('http')]); } elseif (str_starts_with($path, 'preferred-install')) { $types[] = new UnionType([new ConstantStringType('source'), new ConstantStringType('dist'), new ConstantStringType('auto')]); } else { $types[] = new StringType(); } break; case 'boolean': if ($path === 'platform.additionalProperties') { $types[] = new ConstantBooleanType(false); } else { $types[] = new BooleanType(); } break; case 'object': $addlPropType = null; if (isset($def['additionalProperties'])) { $addlPropType = $this->parseType($def['additionalProperties'], $path.'.additionalProperties'); } if (isset($def['properties'])) { $keyNames = []; $valTypes = []; $optionalKeys = []; $propIndex = 0; foreach ($def['properties'] as $propName => $propdef) { $keyNames[] = new ConstantStringType($propName); $valType = $this->parseType($propdef, $path.'.'.$propName); if (!isset($def['required']) || !in_array($propName, $def['required'], true)) { $valType = TypeCombinator::addNull($valType); $optionalKeys[] = $propIndex; } $valTypes[] = $valType; $propIndex++; } if ($addlPropType !== null) { $types[] = new ArrayType(TypeCombinator::union(new StringType(), ...$keyNames), TypeCombinator::union($addlPropType, ...$valTypes)); } else { $types[] = new ConstantArrayType($keyNames, $valTypes, [0], $optionalKeys); } } else { $types[] = new ArrayType(new StringType(), $addlPropType ?? new MixedType()); } break; case 'array': if (isset($def['items'])) { $valType = $this->parseType($def['items'], $path.'.items'); } else { $valType = new MixedType(); } $types[] = new ArrayType(new IntegerType(), $valType); break; default: $types[] = new MixedType(); } } $type = TypeCombinator::union(...$types); } elseif (isset($def['enum'])) { $type = TypeCombinator::union(...array_map(static function (string $value): ConstantStringType { return new ConstantStringType($value); }, $def['enum'])); } else { $type = new MixedType(); } // allow-plugins defaults to null until July 1st 2022 for some BC hackery, but after that it is not nullable anymore if ($path === 'allow-plugins' && time() < strtotime('2022-07-01')) { $type = TypeCombinator::addNull($type); } // default null props if (in_array($path, ['autoloader-suffix', 'gitlab-protocol'], true)) { $type = TypeCombinator::addNull($type); } return $type; } } __halt_compiler();----SIGNATURE:----OxH1uB/aY85z7n98NvK6Cy1/RyHoXPgRMstFmSYdmEgrvLay+hihIRdhRM8Z2n1+QfARf8eXmtUqd+YAWubwMKouEj4+YMoyRRZp7L1GtbMymPC4EjWHOkWgCRrHsE/ZTtCDR7tzb4T3uLBTD+/ZvISj5kZyuA7Z5/AHMV56nlc7fEB4kI9G8zqbYobPtNemZEkFJCKtgoQlet3SJ6KTpReUNvkBePSZYEMmVwOy1so4oXcICYmOAiXcesWcKEYI4Pv4u/0+HwP/TiwQru5hMEB2QPhezdxsIgxmfTNcpg/vuBRil1sYmKf7Pyo4KbCAP895Rkzrm3YRwzmYnwOOZQeKz/igrVZrjCWHielEBweWJekmE0/AgLUbpAnKOlYZPsxmWcr40hGDUY76Z/jahD3wNEHvewaBq22QOZe+IX2p94opz3/VfNvnQ8llMlSQnIK5UH1bYDjL9qRTrYchuUfCibdST+huHSPIcSj60bKQjelfVMfHRm/H/YPvszPnXqXPHcb5Havzg60YLfFEFAAo/lpMnI0P0XvjuiiaPgaYFcIZYw1TRrqtYAaxCUSkFQXfYZEig5F4hScXI3aRY2JI4ozEbl6K6EV9ePm2CMXDaWSl27c4aEdgFGTj+fXvGNKMx2Z5Ar7mWRD8GCRezJ5c7RQ/2mPRLrkUkxJJGrc=----ATTACHMENT:----OTUyMjg0ODA3OTQwMDE3NyA3OTcxODExOTA5MjUyNDA3IDMwNzI4NzU0MzQwODQzOTY=