Skip to content

Commit

Permalink
Revert "Use composer.json reading just to locate functions"
Browse files Browse the repository at this point in the history
This reverts commit 4cbb589.
  • Loading branch information
ondrejmirtes committed Jun 24, 2020
1 parent 1c63a78 commit 37356d4
Show file tree
Hide file tree
Showing 5 changed files with 196 additions and 9 deletions.
3 changes: 3 additions & 0 deletions conf/config.neon
Original file line number Diff line number Diff line change
Expand Up @@ -499,6 +499,9 @@ services:
-
class: PHPStan\Reflection\BetterReflection\SourceLocator\OptimizedDirectorySourceLocatorRepository

-
implement: PHPStan\Reflection\BetterReflection\SourceLocator\OptimizedPsrAutoloaderLocatorFactory

-
implement: PHPStan\Reflection\BetterReflection\SourceLocator\OptimizedSingleFileSourceLocatorFactory

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,14 @@ public function create(): SourceLocator
$locators[] = $this->optimizedSingleFileSourceLocatorRepository->getOrCreate($this->singleReflectionFile);
}

foreach ($this->composerAutoloaderProjectPaths as $composerAutoloaderProjectPath) {
$locator = $this->composerJsonAndInstalledJsonSourceLocatorMaker->create($composerAutoloaderProjectPath);
if ($locator === null) {
continue;
}
$locators[] = $locator;
}

$analysedDirectories = [];
$analysedFiles = [];

Expand Down Expand Up @@ -156,13 +164,6 @@ public function create(): SourceLocator
});
$locators[] = new SkipClassAliasSourceLocator(new PhpInternalSourceLocator($astLocator, $this->phpstormStubsSourceStubber));
$locators[] = $this->autoloadSourceLocator;
foreach ($this->composerAutoloaderProjectPaths as $composerAutoloaderProjectPath) {
$locator = $this->composerJsonAndInstalledJsonSourceLocatorMaker->create($composerAutoloaderProjectPath);
if ($locator === null) {
continue;
}
$locators[] = $locator;
}
$locators[] = new PhpInternalSourceLocator($astLocator, $this->reflectionSourceStubber);
$locators[] = new EvaledCodeSourceLocator($astLocator, $this->reflectionSourceStubber);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,18 +5,28 @@
use Nette\Utils\Json;
use PHPStan\File\FileReader;
use Roave\BetterReflection\SourceLocator\Type\AggregateSourceLocator;
use Roave\BetterReflection\SourceLocator\Type\Composer\Psr\Psr0Mapping;
use Roave\BetterReflection\SourceLocator\Type\Composer\Psr\Psr4Mapping;
use Roave\BetterReflection\SourceLocator\Type\SourceLocator;

class ComposerJsonAndInstalledJsonSourceLocatorMaker
{

private \PHPStan\Reflection\BetterReflection\SourceLocator\OptimizedDirectorySourceLocatorRepository $optimizedDirectorySourceLocatorRepository;

private \PHPStan\Reflection\BetterReflection\SourceLocator\OptimizedSingleFileSourceLocatorRepository $optimizedSingleFileSourceLocatorRepository;

private \PHPStan\Reflection\BetterReflection\SourceLocator\OptimizedPsrAutoloaderLocatorFactory $optimizedPsrAutoloaderLocatorFactory;

public function __construct(
OptimizedSingleFileSourceLocatorRepository $optimizedSingleFileSourceLocatorRepository
OptimizedDirectorySourceLocatorRepository $optimizedDirectorySourceLocatorRepository,
OptimizedSingleFileSourceLocatorRepository $optimizedSingleFileSourceLocatorRepository,
OptimizedPsrAutoloaderLocatorFactory $optimizedPsrAutoloaderLocatorFactory
)
{
$this->optimizedDirectorySourceLocatorRepository = $optimizedDirectorySourceLocatorRepository;
$this->optimizedSingleFileSourceLocatorRepository = $optimizedSingleFileSourceLocatorRepository;
$this->optimizedPsrAutoloaderLocatorFactory = $optimizedPsrAutoloaderLocatorFactory;
}

public function create(string $projectInstallationPath): ?SourceLocator
Expand Down Expand Up @@ -48,6 +58,17 @@ public function create(string $projectInstallationPath): ?SourceLocator

$installed = $installedJson['packages'] ?? $installedJson;

$classMapPaths = array_merge(
$this->prefixPaths($this->packageToClassMapPaths($composer), $projectInstallationPath . '/'),
...array_map(function (array $package) use ($projectInstallationPath, $installedJsonDirectoryPath): array {
return $this->prefixPaths(
$this->packageToClassMapPaths($package),
$this->packagePrefixPath($projectInstallationPath, $installedJsonDirectoryPath, $package)
);
}, $installed)
);
$classMapFiles = array_filter($classMapPaths, 'is_file');
$classMapDirectories = array_filter($classMapPaths, 'is_dir');
$filePaths = array_merge(
$this->prefixPaths($this->packageToFilePaths($composer), $projectInstallationPath . '/'),
...array_map(function (array $package) use ($projectInstallationPath, $installedJsonDirectoryPath): array {
Expand All @@ -59,7 +80,42 @@ public function create(string $projectInstallationPath): ?SourceLocator
);

$locators = [];
foreach ($filePaths as $file) {
$locators[] = $this->optimizedPsrAutoloaderLocatorFactory->create(
Psr4Mapping::fromArrayMappings(array_merge_recursive(
$this->prefixWithInstallationPath($this->packageToPsr4AutoloadNamespaces($composer), $projectInstallationPath),
...array_map(function (array $package) use ($projectInstallationPath, $installedJsonDirectoryPath): array {
return $this->prefixWithPackagePath(
$this->packageToPsr4AutoloadNamespaces($package),
$projectInstallationPath,
$installedJsonDirectoryPath,
$package
);
}, $installed)
))
);

$locators[] = $this->optimizedPsrAutoloaderLocatorFactory->create(
Psr0Mapping::fromArrayMappings(array_merge_recursive(
$this->prefixWithInstallationPath($this->packageToPsr0AutoloadNamespaces($composer), $projectInstallationPath),
...array_map(function (array $package) use ($projectInstallationPath, $installedJsonDirectoryPath): array {
return $this->prefixWithPackagePath(
$this->packageToPsr0AutoloadNamespaces($package),
$projectInstallationPath,
$installedJsonDirectoryPath,
$package
);
}, $installed)
))
);

foreach ($classMapDirectories as $classMapDirectory) {
if (!is_dir($classMapDirectory)) {
continue;
}
$locators[] = $this->optimizedDirectorySourceLocatorRepository->getOrCreate($classMapDirectory);
}

foreach (array_merge($classMapFiles, $filePaths) as $file) {
if (!is_file($file)) {
continue;
}
Expand All @@ -69,6 +125,40 @@ public function create(string $projectInstallationPath): ?SourceLocator
return new AggregateSourceLocator($locators);
}

/**
* @param mixed[] $package
*
* @return array<string, array<int, string>>
*/
private function packageToPsr4AutoloadNamespaces(array $package): array
{
return array_map(static function ($namespacePaths): array {
return (array) $namespacePaths;
}, $package['autoload']['psr-4'] ?? []);
}

/**
* @param mixed[] $package
*
* @return array<string, array<int, string>>
*/
private function packageToPsr0AutoloadNamespaces(array $package): array
{
return array_map(static function ($namespacePaths): array {
return (array) $namespacePaths;
}, $package['autoload']['psr-0'] ?? []);
}

/**
* @param mixed[] $package
*
* @return array<int, string>
*/
private function packageToClassMapPaths(array $package): array
{
return $package['autoload']['classmap'] ?? [];
}

/**
* @param mixed[] $package
*
Expand All @@ -95,6 +185,33 @@ private function packagePrefixPath(
return $projectInstallationPath . '/vendor/' . $package['name'] . '/';
}

/**
* @param array<string, array<int, string>> $paths
* @param array<string, array<int, string>> $package
*
* @return array<string, array<int, string>>
*/
private function prefixWithPackagePath(array $paths, string $projectInstallationPath, string $installedJsonDirectoryPath, array $package): array
{
$prefix = $this->packagePrefixPath($projectInstallationPath, $installedJsonDirectoryPath, $package);

return array_map(function (array $paths) use ($prefix): array {
return $this->prefixPaths($paths, $prefix);
}, $paths);
}

/**
* @param array<int|string, array<string>> $paths
*
* @return array<int|string, array<string>>
*/
private function prefixWithInstallationPath(array $paths, string $trimmedInstallationPath): array
{
return array_map(function (array $paths) use ($trimmedInstallationPath): array {
return $this->prefixPaths($paths, $trimmedInstallationPath . '/');
}, $paths);
}

/**
* @param array<int, string> $paths
*
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
<?php declare(strict_types = 1);

namespace PHPStan\Reflection\BetterReflection\SourceLocator;

use Roave\BetterReflection\Identifier\Identifier;
use Roave\BetterReflection\Identifier\IdentifierType;
use Roave\BetterReflection\Reflection\Reflection;
use Roave\BetterReflection\Reflector\Reflector;
use Roave\BetterReflection\SourceLocator\Type\Composer\Psr\PsrAutoloaderMapping;
use Roave\BetterReflection\SourceLocator\Type\SourceLocator;

class OptimizedPsrAutoloaderLocator implements SourceLocator
{

private PsrAutoloaderMapping $mapping;

private \PHPStan\Reflection\BetterReflection\SourceLocator\OptimizedSingleFileSourceLocatorRepository $optimizedSingleFileSourceLocatorRepository;

public function __construct(
PsrAutoloaderMapping $mapping,
OptimizedSingleFileSourceLocatorRepository $optimizedSingleFileSourceLocatorRepository
)
{
$this->mapping = $mapping;
$this->optimizedSingleFileSourceLocatorRepository = $optimizedSingleFileSourceLocatorRepository;
}

public function locateIdentifier(Reflector $reflector, Identifier $identifier): ?Reflection
{
foreach ($this->mapping->resolvePossibleFilePaths($identifier) as $file) {
if (!file_exists($file)) {
continue;
}

$reflection = $this->optimizedSingleFileSourceLocatorRepository->getOrCreate($file)->locateIdentifier($reflector, $identifier);
if ($reflection === null) {
continue;
}

return $reflection;
}

return null;
}

/**
* @return Reflection[]
*/
public function locateIdentifiersByType(Reflector $reflector, IdentifierType $identifierType): array
{
return []; // todo
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
<?php declare(strict_types = 1);

namespace PHPStan\Reflection\BetterReflection\SourceLocator;

use Roave\BetterReflection\SourceLocator\Type\Composer\Psr\PsrAutoloaderMapping;

interface OptimizedPsrAutoloaderLocatorFactory
{

public function create(PsrAutoloaderMapping $mapping): OptimizedPsrAutoloaderLocator;

}

0 comments on commit 37356d4

Please sign in to comment.