Code Coverage |
||||||||||
Lines |
Functions and Methods |
Classes and Traits |
||||||||
Total | |
67.57% |
25 / 37 |
|
44.44% |
4 / 9 |
CRAP | |
0.00% |
0 / 1 |
ComposerPackage | |
67.57% |
25 / 37 |
|
44.44% |
4 / 9 |
29.05 | |
0.00% |
0 / 1 |
fromFile | |
100.00% |
2 / 2 |
|
100.00% |
1 / 1 |
1 | |||
fromComposerJsonArray | |
0.00% |
0 / 4 |
|
0.00% |
0 / 1 |
2 | |||
__construct | |
77.27% |
17 / 22 |
|
0.00% |
0 / 1 |
9.95 | |||
getPackageName | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
1 | |||
getRelativePath | |
0.00% |
0 / 1 |
|
0.00% |
0 / 1 |
2 | |||
getPackageAbsolutePath | |
0.00% |
0 / 1 |
|
0.00% |
0 / 1 |
2 | |||
getAutoload | |
100.00% |
1 / 1 |
|
100.00% |
1 / 1 |
1 | |||
getRequiresNames | |
100.00% |
4 / 4 |
|
100.00% |
1 / 1 |
2 | |||
getLicense | |
0.00% |
0 / 1 |
|
0.00% |
0 / 1 |
2 |
1 | <?php |
2 | /** |
3 | * Object for getting typed values from composer.json. |
4 | * |
5 | * Use this for dependencies. Use ProjectComposerPackage for the primary composer.json. |
6 | */ |
7 | |
8 | namespace BrianHenryIE\Strauss\Composer; |
9 | |
10 | use Composer\Composer; |
11 | use Composer\Factory; |
12 | use Composer\IO\NullIO; |
13 | |
14 | /** |
15 | * @phpstan-type AutoloadKey array{files?:array<string>,classmap?:array<string>,"psr-4"?:array<string,string|array<string>>} |
16 | */ |
17 | class ComposerPackage |
18 | { |
19 | /** |
20 | * The composer.json file as parsed by Composer. |
21 | * |
22 | * @see \Composer\Factory::create |
23 | * |
24 | * @var \Composer\Composer |
25 | */ |
26 | protected \Composer\Composer $composer; |
27 | |
28 | /** |
29 | * The name of the project in composer.json. |
30 | * |
31 | * e.g. brianhenryie/my-project |
32 | * |
33 | * @var string |
34 | */ |
35 | protected string $packageName; |
36 | |
37 | /** |
38 | * Virtual packages and meta packages do not have a composer.json. |
39 | * Some packages are installed in a different directory name than their package name. |
40 | * |
41 | * @var ?string |
42 | */ |
43 | protected ?string $relativePath = null; |
44 | |
45 | /** |
46 | * Packages can be symlinked from outside the current project directory. |
47 | * |
48 | * @var ?string |
49 | */ |
50 | protected ?string $packageAbsolutePath = null; |
51 | |
52 | /** |
53 | * The discovered files, classmap, psr0 and psr4 autoload keys discovered (as parsed by Composer). |
54 | * |
55 | * @var AutoloadKey |
56 | */ |
57 | protected array $autoload = []; |
58 | |
59 | /** |
60 | * The names in the composer.json's "requires" field (without versions). |
61 | * |
62 | * @var string[] |
63 | */ |
64 | protected array $requiresNames = []; |
65 | |
66 | protected string $license; |
67 | |
68 | /** |
69 | * @param string $absolutePath The absolute path to composer.json |
70 | * @param ?array{files?:array<string>, classmap?:array<string>, psr?:array<string,string|array<string>>} $overrideAutoload Optional configuration to replace the package's own autoload definition with |
71 | * another which Strauss can use. |
72 | * @return ComposerPackage |
73 | */ |
74 | public static function fromFile(string $absolutePath, array $overrideAutoload = null): ComposerPackage |
75 | { |
76 | $composer = Factory::create(new NullIO(), $absolutePath, true); |
77 | |
78 | return new ComposerPackage($composer, $overrideAutoload); |
79 | } |
80 | |
81 | /** |
82 | * This is used for virtual packages, which don't have a composer.json. |
83 | * |
84 | * @param array{name?:string, license?:string, requires?:array<string,string>, autoload?:AutoloadKey} $jsonArray composer.json decoded to array |
85 | * @param ?AutoloadKey $overrideAutoload New autoload rules to replace the existing ones. |
86 | */ |
87 | public static function fromComposerJsonArray($jsonArray, array $overrideAutoload = null): ComposerPackage |
88 | { |
89 | $factory = new Factory(); |
90 | $io = new NullIO(); |
91 | $composer = $factory->createComposer($io, $jsonArray, true); |
92 | |
93 | return new ComposerPackage($composer, $overrideAutoload); |
94 | } |
95 | |
96 | /** |
97 | * Create a PHP object to represent a composer package. |
98 | * |
99 | * @param Composer $composer |
100 | * @param ?AutoloadKey $overrideAutoload Optional configuration to replace the package's own autoload definition with another which Strauss can use. |
101 | */ |
102 | public function __construct(Composer $composer, array $overrideAutoload = null) |
103 | { |
104 | $this->composer = $composer; |
105 | |
106 | $this->packageName = $composer->getPackage()->getName(); |
107 | |
108 | $composerJsonFileAbsolute = $composer->getConfig()->getConfigSource()->getName(); |
109 | |
110 | $absolutePath = realpath(dirname($composerJsonFileAbsolute)); |
111 | if (false !== $absolutePath) { |
112 | $this->packageAbsolutePath = $absolutePath . '/'; |
113 | } |
114 | |
115 | $vendorDirectory = $this->composer->getConfig()->get('vendor-dir'); |
116 | if (file_exists($vendorDirectory . '/' . $this->packageName)) { |
117 | $this->relativePath = $this->packageName; |
118 | $this->packageAbsolutePath = realpath($vendorDirectory . '/' . $this->packageName) . '/'; |
119 | // If the package is symlinked, the path will be outside the working directory. |
120 | } elseif (0 !== strpos($absolutePath, getcwd()) && 1 === preg_match('/.*[\/\\\\]([^\/\\\\]*[\/\\\\][^\/\\\\]*)[\/\\\\][^\/\\\\]*/', $vendorDirectory, $output_array)) { |
121 | $this->relativePath = $output_array[1]; |
122 | } elseif (1 === preg_match('/.*[\/\\\\]([^\/\\\\]+[\/\\\\][^\/\\\\]+)[\/\\\\]composer.json/', $composerJsonFileAbsolute, $output_array)) { |
123 | // Not every package gets installed to a folder matching its name (crewlabs/unsplash). |
124 | $this->relativePath = $output_array[1]; |
125 | } |
126 | |
127 | if (!is_null($overrideAutoload)) { |
128 | $composer->getPackage()->setAutoload($overrideAutoload); |
129 | } |
130 | |
131 | $this->autoload = $composer->getPackage()->getAutoload(); |
132 | |
133 | foreach ($composer->getPackage()->getRequires() as $_name => $packageLink) { |
134 | $this->requiresNames[] = $packageLink->getTarget(); |
135 | } |
136 | |
137 | // Try to get the license from the package's composer.json, assume proprietary (all rights reserved!). |
138 | $this->license = !empty($composer->getPackage()->getLicense()) |
139 | ? implode(',', $composer->getPackage()->getLicense()) |
140 | : 'proprietary?'; |
141 | } |
142 | |
143 | /** |
144 | * Composer package project name. |
145 | * |
146 | * vendor/project-name |
147 | * |
148 | * @return string |
149 | */ |
150 | public function getPackageName(): string |
151 | { |
152 | return $this->packageName; |
153 | } |
154 | |
155 | /** |
156 | * Is this relative to vendor? |
157 | */ |
158 | public function getRelativePath(): ?string |
159 | { |
160 | return $this->relativePath . '/'; |
161 | } |
162 | |
163 | public function getPackageAbsolutePath(): ?string |
164 | { |
165 | return $this->packageAbsolutePath; |
166 | } |
167 | |
168 | /** |
169 | * |
170 | * e.g. ['psr-4' => [ 'BrianHenryIE\Project' => 'src' ]] |
171 | * e.g. ['psr-4' => [ 'BrianHenryIE\Project' => ['src','lib] ]] |
172 | * e.g. ['classmap' => [ 'src', 'lib' ]] |
173 | * e.g. ['files' => [ 'lib', 'functions.php' ]] |
174 | * |
175 | * @return AutoloadKey |
176 | */ |
177 | public function getAutoload(): array |
178 | { |
179 | return $this->autoload; |
180 | } |
181 | |
182 | /** |
183 | * The names of the packages in the composer.json's "requires" field (without version). |
184 | * |
185 | * Excludes PHP, ext-*, since we won't be copying or prefixing them. |
186 | * |
187 | * @return string[] |
188 | */ |
189 | public function getRequiresNames(): array |
190 | { |
191 | // Unset PHP, ext-*. |
192 | $removePhpExt = function ($element) { |
193 | return !( 0 === strpos($element, 'ext') || 'php' === $element ); |
194 | }; |
195 | |
196 | return array_filter($this->requiresNames, $removePhpExt); |
197 | } |
198 | |
199 | public function getLicense():string |
200 | { |
201 | return $this->license; |
202 | } |
203 | } |