Code Coverage |
||||||||||
Lines |
Functions and Methods |
Classes and Traits |
||||||||
| Total | |
41.56% |
64 / 154 |
|
28.57% |
2 / 7 |
CRAP | |
0.00% |
0 / 1 |
| VendorComposerAutoload | |
41.56% |
64 / 154 |
|
28.57% |
2 / 7 |
160.93 | |
0.00% |
0 / 1 |
| __construct | |
100.00% |
3 / 3 |
|
100.00% |
1 / 1 |
1 | |||
| addVendorPrefixedAutoloadToVendorAutoload | |
0.00% |
0 / 17 |
|
0.00% |
0 / 1 |
30 | |||
| addAliasesFileToComposer | |
58.82% |
10 / 17 |
|
0.00% |
0 / 1 |
6.75 | |||
| isComposerInstalled | |
75.00% |
3 / 4 |
|
0.00% |
0 / 1 |
2.06 | |||
| isComposerNoDev | |
100.00% |
3 / 3 |
|
100.00% |
1 / 1 |
1 | |||
| addAliasesFileToComposerAutoload | |
90.00% |
45 / 50 |
|
0.00% |
0 / 1 |
5.03 | |||
| addVendorPrefixedAutoloadToComposerAutoload | |
0.00% |
0 / 60 |
|
0.00% |
0 / 1 |
56 | |||
| 1 | <?php |
| 2 | /** |
| 3 | * Edit vendor/autoload.php to also load the vendor/composer/autoload_aliases.php file and the vendor-prefixed/autoload.php file. |
| 4 | */ |
| 5 | |
| 6 | namespace BrianHenryIE\Strauss\Pipeline\Autoload; |
| 7 | |
| 8 | use BrianHenryIE\Strauss\Composer\Extra\StraussConfig; |
| 9 | use BrianHenryIE\Strauss\Config\AutoloadConfigInterface; |
| 10 | use BrianHenryIE\Strauss\Helpers\FileSystem; |
| 11 | use PhpParser\Error; |
| 12 | use PhpParser\Node; |
| 13 | use PhpParser\NodeTraverser; |
| 14 | use PhpParser\NodeVisitorAbstract; |
| 15 | use PhpParser\ParserFactory; |
| 16 | use PhpParser\PrettyPrinter\Standard; |
| 17 | use Psr\Log\LoggerAwareTrait; |
| 18 | use Psr\Log\LoggerInterface; |
| 19 | |
| 20 | class VendorComposerAutoload |
| 21 | { |
| 22 | use LoggerAwareTrait; |
| 23 | |
| 24 | protected FileSystem $fileSystem; |
| 25 | |
| 26 | protected AutoloadConfigInterface $config; |
| 27 | |
| 28 | public function __construct( |
| 29 | AutoloadConfigInterface $config, |
| 30 | Filesystem $filesystem, |
| 31 | LoggerInterface $logger |
| 32 | ) { |
| 33 | $this->config = $config; |
| 34 | $this->fileSystem = $filesystem; |
| 35 | $this->setLogger($logger); |
| 36 | } |
| 37 | |
| 38 | public function addVendorPrefixedAutoloadToVendorAutoload(): void |
| 39 | { |
| 40 | if ($this->config->getTargetDirectory() === $this->config->getVendorDirectory()) { |
| 41 | $this->logger->info("Target dir is source dir, no autoload.php to add."); |
| 42 | return; |
| 43 | } |
| 44 | |
| 45 | $composerAutoloadPhpFilepath = $this->config->getVendorDirectory() . 'autoload.php'; |
| 46 | |
| 47 | if (!$this->fileSystem->fileExists($composerAutoloadPhpFilepath)) { |
| 48 | $this->logger->info("No autoload.php found:" . $composerAutoloadPhpFilepath); |
| 49 | return; |
| 50 | } |
| 51 | |
| 52 | $newAutoloadPhpFilepath = $this->config->getTargetDirectory() . 'autoload.php'; |
| 53 | |
| 54 | if (!$this->fileSystem->fileExists($newAutoloadPhpFilepath)) { |
| 55 | $this->logger->warning("No new autoload.php found: " . $newAutoloadPhpFilepath); |
| 56 | } |
| 57 | |
| 58 | $this->logger->info('Modifying original autoload.php to add `' . $newAutoloadPhpFilepath); |
| 59 | |
| 60 | $composerAutoloadPhpFileString = $this->fileSystem->read($composerAutoloadPhpFilepath); |
| 61 | |
| 62 | $newComposerAutoloadPhpFileString = $this->addVendorPrefixedAutoloadToComposerAutoload($composerAutoloadPhpFileString); |
| 63 | |
| 64 | if ($newComposerAutoloadPhpFileString !== $composerAutoloadPhpFileString) { |
| 65 | $this->logger->info('Writing new autoload.php'); |
| 66 | $this->fileSystem->write($composerAutoloadPhpFilepath, $newComposerAutoloadPhpFileString); |
| 67 | } else { |
| 68 | $this->logger->debug('No changes to autoload.php'); |
| 69 | } |
| 70 | } |
| 71 | |
| 72 | /** |
| 73 | * Given the PHP code string for `vendor/autoload.php`, add a `require_once autoload_aliases.php` |
| 74 | * before require autoload_real.php. |
| 75 | */ |
| 76 | public function addAliasesFileToComposer(): void |
| 77 | { |
| 78 | if ($this->isComposerInstalled()) { |
| 79 | $this->logger->info("Strauss installed via Composer, no need to add `autoload_aliases.php` to `vendor/autoload.php`"); |
| 80 | return; |
| 81 | } |
| 82 | |
| 83 | $composerAutoloadPhpFilepath = $this->config->getVendorDirectory() . 'autoload.php'; |
| 84 | |
| 85 | if (!$this->fileSystem->fileExists($composerAutoloadPhpFilepath)) { |
| 86 | // No `vendor/autoload.php` file to add `autoload_aliases.php` to. |
| 87 | $this->logger->error("No autoload.php found: " . $composerAutoloadPhpFilepath); |
| 88 | // TODO: Should probably throw an exception here. |
| 89 | return; |
| 90 | } |
| 91 | |
| 92 | if ($this->isComposerNoDev()) { |
| 93 | $this->logger->notice("Composer was run with `--no-dev`, no need to add `autoload_aliases.php` to `vendor/autoload.php`"); |
| 94 | return; |
| 95 | } |
| 96 | |
| 97 | $this->logger->info('Modifying original autoload.php to add autoload_aliases.php in ' . $this->config->getVendorDirectory()); |
| 98 | |
| 99 | $composerAutoloadPhpFileString = $this->fileSystem->read($composerAutoloadPhpFilepath); |
| 100 | |
| 101 | $newComposerAutoloadPhpFileString = $this->addAliasesFileToComposerAutoload($composerAutoloadPhpFileString); |
| 102 | |
| 103 | if ($newComposerAutoloadPhpFileString !== $composerAutoloadPhpFileString) { |
| 104 | $this->logger->info('Writing new autoload.php'); |
| 105 | $this->fileSystem->write($composerAutoloadPhpFilepath, $newComposerAutoloadPhpFileString); |
| 106 | } else { |
| 107 | $this->logger->debug('No changes to autoload.php'); |
| 108 | } |
| 109 | } |
| 110 | |
| 111 | /** |
| 112 | * Determine is Strauss installed via Composer (otherwise presumably run via phar). |
| 113 | */ |
| 114 | protected function isComposerInstalled(): bool |
| 115 | { |
| 116 | if (!$this->fileSystem->fileExists($this->config->getVendorDirectory() . 'composer/installed.json')) { |
| 117 | return false; |
| 118 | } |
| 119 | |
| 120 | $installedJsonArray = json_decode($this->fileSystem->read($this->config->getVendorDirectory() . 'composer/installed.json'), true); |
| 121 | |
| 122 | return isset($installedJsonArray['dev-package-names']['brianhenryie/strauss']); |
| 123 | } |
| 124 | |
| 125 | /** |
| 126 | * Read `vendor/composer/installed.json` to determine if the composer was run with `--no-dev`. |
| 127 | * |
| 128 | * { |
| 129 | * "packages": [], |
| 130 | * "dev": true, |
| 131 | * "dev-package-names": [] |
| 132 | * } |
| 133 | */ |
| 134 | protected function isComposerNoDev(): bool |
| 135 | { |
| 136 | $installedJson = $this->fileSystem->read($this->config->getVendorDirectory() . 'composer/installed.json'); |
| 137 | $installedJsonArray = json_decode($installedJson, true); |
| 138 | return !$installedJsonArray['dev']; |
| 139 | } |
| 140 | |
| 141 | /** |
| 142 | * This is a very over-engineered way to do a string replace. |
| 143 | * |
| 144 | * `require_once __DIR__ . '/composer/autoload_aliases.php';` |
| 145 | */ |
| 146 | protected function addAliasesFileToComposerAutoload(string $code): string |
| 147 | { |
| 148 | if (false !== strpos($code, '/composer/autoload_aliases.php')) { |
| 149 | $this->logger->info('vendor/autoload.php already includes autoload_aliases.php'); |
| 150 | return $code; |
| 151 | } |
| 152 | |
| 153 | $parser = (new ParserFactory())->createForNewestSupportedVersion(); |
| 154 | try { |
| 155 | $ast = $parser->parse($code); |
| 156 | } catch (Error $error) { |
| 157 | $this->logger->error("Parse error: {$error->getMessage()}"); |
| 158 | return $code; |
| 159 | } |
| 160 | |
| 161 | $traverser = new NodeTraverser(); |
| 162 | $traverser->addVisitor(new class() extends NodeVisitorAbstract { |
| 163 | |
| 164 | public function leaveNode(Node $node) |
| 165 | { |
| 166 | if (get_class($node) === \PhpParser\Node\Stmt\Expression::class) { |
| 167 | $prettyPrinter = new Standard(); |
| 168 | $maybeRequireAutoloadReal = $prettyPrinter->prettyPrintExpr($node->expr); |
| 169 | |
| 170 | // Every `vendor/autoload.php` should have this line. |
| 171 | $target = "require_once __DIR__ . '/composer/autoload_real.php'"; |
| 172 | |
| 173 | // If this node isn't the one we want to insert before, continue. |
| 174 | if ($maybeRequireAutoloadReal !== $target) { |
| 175 | return $node; |
| 176 | } |
| 177 | |
| 178 | // __DIR__ . '/composer/autoload_aliases.php' |
| 179 | $path = new \PhpParser\Node\Expr\BinaryOp\Concat( |
| 180 | new \PhpParser\Node\Scalar\MagicConst\Dir(), |
| 181 | new \PhpParser\Node\Scalar\String_('/composer/autoload_aliases.php') |
| 182 | ); |
| 183 | |
| 184 | // require_once |
| 185 | $requireOnceAutoloadAliases = new Node\Stmt\Expression( |
| 186 | new \PhpParser\Node\Expr\Include_( |
| 187 | $path, |
| 188 | \PhpParser\Node\Expr\Include_::TYPE_REQUIRE_ONCE |
| 189 | ) |
| 190 | ); |
| 191 | |
| 192 | // if(file_exists()){} |
| 193 | $ifFileExistsRequireOnceAutoloadAliases = new \PhpParser\Node\Stmt\If_( |
| 194 | new \PhpParser\Node\Expr\FuncCall( |
| 195 | new \PhpParser\Node\Name('file_exists'), |
| 196 | [ |
| 197 | new \PhpParser\Node\Arg($path) |
| 198 | ], |
| 199 | ), |
| 200 | [ |
| 201 | 'stmts' => [ |
| 202 | $requireOnceAutoloadAliases |
| 203 | ], |
| 204 | ] |
| 205 | ); |
| 206 | |
| 207 | // Add a blank line. Probably not the correct way to do this. |
| 208 | $node->setAttribute('comments', [new \PhpParser\Comment('')]); |
| 209 | $ifFileExistsRequireOnceAutoloadAliases->setAttribute('comments', [new \PhpParser\Comment('')]); |
| 210 | |
| 211 | return [ |
| 212 | $ifFileExistsRequireOnceAutoloadAliases, |
| 213 | $node |
| 214 | ]; |
| 215 | } |
| 216 | return $node; |
| 217 | } |
| 218 | }); |
| 219 | |
| 220 | $modifiedStmts = $traverser->traverse($ast); |
| 221 | |
| 222 | $prettyPrinter = new Standard(); |
| 223 | |
| 224 | return $prettyPrinter->prettyPrintFile($modifiedStmts); |
| 225 | } |
| 226 | |
| 227 | /** |
| 228 | * `require_once __DIR__ . '/../vendor-prefixed/autoload.php';` |
| 229 | */ |
| 230 | protected function addVendorPrefixedAutoloadToComposerAutoload(string $code): string |
| 231 | { |
| 232 | if ($this->config->getTargetDirectory() === $this->config->getVendorDirectory()) { |
| 233 | $this->logger->info('Vendor directory is target directory, no autoloader to add.'); |
| 234 | return $code; |
| 235 | } |
| 236 | |
| 237 | $targetDirAutoload = '/' . $this->fileSystem->getRelativePath($this->config->getVendorDirectory(), $this->config->getTargetDirectory()) . 'autoload.php'; |
| 238 | |
| 239 | if (false !== strpos($code, $targetDirAutoload)) { |
| 240 | $this->logger->info('vendor/autoload.php already includes ' . $targetDirAutoload); |
| 241 | return $code; |
| 242 | } |
| 243 | |
| 244 | $parser = (new ParserFactory())->createForNewestSupportedVersion(); |
| 245 | try { |
| 246 | $ast = $parser->parse($code); |
| 247 | } catch (Error $error) { |
| 248 | $this->logger->error("Parse error: {$error->getMessage()}"); |
| 249 | return $code; |
| 250 | } |
| 251 | |
| 252 | $traverser = new NodeTraverser(); |
| 253 | $traverser->addVisitor(new class($targetDirAutoload) extends NodeVisitorAbstract { |
| 254 | |
| 255 | protected bool $added = false; |
| 256 | protected ?string $targetDirectoryAutoload; |
| 257 | public function __construct(?string $targetDirectoryAutoload) |
| 258 | { |
| 259 | $this->targetDirectoryAutoload = $targetDirectoryAutoload; |
| 260 | } |
| 261 | |
| 262 | public function leaveNode(Node $node) |
| 263 | { |
| 264 | if ($this->added) { |
| 265 | return $node; |
| 266 | } |
| 267 | |
| 268 | if (get_class($node) === \PhpParser\Node\Stmt\Expression::class) { |
| 269 | $prettyPrinter = new Standard(); |
| 270 | $nodeText = $prettyPrinter->prettyPrintExpr($node->expr); |
| 271 | |
| 272 | $targets = [ |
| 273 | "require_once __DIR__ . '/composer/autoload_real.php'", |
| 274 | ]; |
| 275 | |
| 276 | if (!in_array($nodeText, $targets)) { |
| 277 | return $node; |
| 278 | } |
| 279 | |
| 280 | // __DIR__ . '../vendor-prefixed/autoload.php' |
| 281 | $path = new \PhpParser\Node\Expr\BinaryOp\Concat( |
| 282 | new \PhpParser\Node\Scalar\MagicConst\Dir(), |
| 283 | new Node\Scalar\String_($this->targetDirectoryAutoload) |
| 284 | ); |
| 285 | |
| 286 | // require_once |
| 287 | $requireOnceStraussAutoload = new Node\Stmt\Expression( |
| 288 | new Node\Expr\Include_( |
| 289 | $path, |
| 290 | Node\Expr\Include_::TYPE_REQUIRE_ONCE |
| 291 | ) |
| 292 | ); |
| 293 | |
| 294 | // if(file_exists()){} |
| 295 | $ifFileExistsRequireOnceStraussAutoload = new \PhpParser\Node\Stmt\If_( |
| 296 | new \PhpParser\Node\Expr\FuncCall( |
| 297 | new \PhpParser\Node\Name('file_exists'), |
| 298 | [ |
| 299 | new \PhpParser\Node\Arg($path) |
| 300 | ], |
| 301 | ), |
| 302 | [ |
| 303 | 'stmts' => [ |
| 304 | $requireOnceStraussAutoload |
| 305 | ], |
| 306 | ] |
| 307 | ); |
| 308 | |
| 309 | // Add a blank line. Probably not the correct way to do this. |
| 310 | $node->setAttribute('comments', [new \PhpParser\Comment('')]); |
| 311 | $ifFileExistsRequireOnceStraussAutoload->setAttribute('comments', [new \PhpParser\Comment('')]); |
| 312 | |
| 313 | $this->added = true; |
| 314 | |
| 315 | return [ |
| 316 | $ifFileExistsRequireOnceStraussAutoload, |
| 317 | $node |
| 318 | ]; |
| 319 | } |
| 320 | return $node; |
| 321 | } |
| 322 | }); |
| 323 | |
| 324 | $modifiedStmts = $traverser->traverse($ast); |
| 325 | |
| 326 | $prettyPrinter = new Standard(); |
| 327 | |
| 328 | return $prettyPrinter->prettyPrintFile($modifiedStmts); |
| 329 | } |
| 330 | } |