Skip to content

Commit 6b227d8

Browse files
Add command to dump required PHP extensions based on vendor/composer/… (#599)
* Add command to dump required PHP extensions based on vendor/composer/installed.json, composer.lock, composer.json (in this order) * remove unused use * missing translation * Adjust dump-extensions * Add docs for dump-extension command --------- Co-authored-by: crazywhalecc <[email protected]>
1 parent 3493436 commit 6b227d8

File tree

6 files changed

+220
-5
lines changed

6 files changed

+220
-5
lines changed

README.md

+2
Original file line numberDiff line numberDiff line change
@@ -194,6 +194,8 @@ Basic usage for building php with some extensions:
194194

195195
# fetch all libraries
196196
./bin/spc download --all
197+
# dump a list of extensions required by your project
198+
./bin/spc dump-extensions
197199
# only fetch necessary sources by needed extensions (recommended)
198200
./bin/spc download --for-extensions="openssl,pcntl,mbstring,pdo_sqlite"
199201
# download pre-built libraries first (save time for compiling dependencies)

docs/en/guide/manual-build.md

+25
Original file line numberDiff line numberDiff line change
@@ -397,6 +397,31 @@ manually unpack and copy the package to a specified location, and we can use com
397397
bin/spc extract php-src,libxml2
398398
```
399399

400+
## Command - dump-extensions
401+
402+
Use the command `bin/spc dump-extensions` to export required extensions of the current project.
403+
404+
```bash
405+
# Print the extension list of the project, pass in the root directory of the project containing composer.json
406+
bin/spc dump-extensions /path/to/your/project/
407+
408+
# Print the extension list of the project, excluding development dependencies
409+
bin/spc dump-extensions /path-to/tour/project/ --no-dev
410+
411+
# Output in the extension list format acceptable to the spc command (comma separated)
412+
bin/spc dump-extensions /path-to/tour/project/ --format=text
413+
414+
# Output as a JSON list
415+
bin/spc dump-extensions /path-to/tour/project/ --format=json
416+
417+
# When the project does not have any extensions, output the specified extension combination instead of returning failure
418+
bin/spc dump-extensions /path-to/your/project/ --no-ext-output=mbstring,posix,pcntl,phar
419+
420+
# Do not exclude extensions not supported by spc when outputting
421+
bin/spc dump-extensions /path/to/your/project/ --no-spc-filter
422+
```
423+
It should be noted that the project directory must contain the `vendor/installed.json` and `composer.lock` files, otherwise they cannot be found normally.
424+
400425
## Dev Command - dev
401426

402427
Debug commands refer to a collection of commands that can assist in outputting some information

docs/zh/guide/manual-build.md

+26
Original file line numberDiff line numberDiff line change
@@ -353,6 +353,32 @@ memory_limit=1G
353353
bin/spc extract php-src,libxml2
354354
```
355355

356+
## 命令 dump-extensions - 导出项目扩展依赖
357+
358+
使用命令 `bin/spc dump-extensions` 可以导出当前项目的扩展依赖。
359+
360+
```bash
361+
# 打印项目的扩展列表,传入项目包含composer.json的根目录
362+
bin/spc dump-extensions /path/to/your/project/
363+
364+
# 打印项目的扩展列表,不包含开发依赖
365+
bin/spc dump-extensions /path-to/tour/project/ --no-dev
366+
367+
# 输出为 spc 命令可接受的扩展列表格式(逗号分割)
368+
bin/spc dump-extensions /path-to/tour/project/ --format=text
369+
370+
# 输出为 JSON 列表
371+
bin/spc dump-extensions /path-to/tour/project/ --format=json
372+
373+
# 当项目没有任何扩展时,输出指定扩展组合,而不是返回失败
374+
bin/spc dump-extensions /path-to/your/project/ --no-ext-output=mbstring,posix,pcntl,phar
375+
376+
# 输出时不排除 spc 不支持的扩展
377+
bin/spc dump-extensions /path/to/your/project/ --no-spc-filter
378+
```
379+
380+
需要注意的是,项目的目录下必须包含 `vendor/installed.json``composer.lock` 文件,否则无法正常获取。
381+
356382
## 调试命令 dev - 调试命令集合
357383

358384
调试命令指的是你在使用 static-php-cli 构建 PHP 或改造、增强 static-php-cli 项目本身的时候,可以辅助输出一些信息的命令集合。

src/SPC/ConsoleApplication.php

+2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
use SPC\command\dev\SortConfigCommand;
1919
use SPC\command\DoctorCommand;
2020
use SPC\command\DownloadCommand;
21+
use SPC\command\DumpExtensionsCommand;
2122
use SPC\command\DumpLicenseCommand;
2223
use SPC\command\ExtractCommand;
2324
use SPC\command\InstallPkgCommand;
@@ -54,6 +55,7 @@ public function __construct()
5455
new MicroCombineCommand(),
5556
new SwitchPhpVersionCommand(),
5657
new SPCConfigCommand(),
58+
new DumpExtensionsCommand(),
5759

5860
// Dev commands
5961
new AllExtCommand(),

src/SPC/command/BaseCommand.php

+5-5
Original file line numberDiff line numberDiff line change
@@ -154,24 +154,24 @@ protected function logWithResult(bool $result, string $success_msg, string $fail
154154
/**
155155
* Parse extension list from string, replace alias and filter internal extensions.
156156
*
157-
* @param string $ext_list Extension string list, e.g. "mbstring,posix,sockets"
157+
* @param array|string $ext_list Extension string list, e.g. "mbstring,posix,sockets" or array
158158
*/
159-
protected function parseExtensionList(string $ext_list): array
159+
protected function parseExtensionList(array|string $ext_list): array
160160
{
161161
// replace alias
162162
$ls = array_map(function ($x) {
163163
$lower = strtolower(trim($x));
164164
if (isset(SPC_EXTENSION_ALIAS[$lower])) {
165-
logger()->notice("Extension [{$lower}] is an alias of [" . SPC_EXTENSION_ALIAS[$lower] . '], it will be replaced.');
165+
logger()->debug("Extension [{$lower}] is an alias of [" . SPC_EXTENSION_ALIAS[$lower] . '], it will be replaced.');
166166
return SPC_EXTENSION_ALIAS[$lower];
167167
}
168168
return $lower;
169-
}, explode(',', $ext_list));
169+
}, is_array($ext_list) ? $ext_list : explode(',', $ext_list));
170170

171171
// filter internals
172172
return array_values(array_filter($ls, function ($x) {
173173
if (in_array($x, SPC_INTERNAL_EXTENSIONS)) {
174-
logger()->warning("Extension [{$x}] is an builtin extension, it will be ignored.");
174+
logger()->debug("Extension [{$x}] is an builtin extension, it will be ignored.");
175175
return false;
176176
}
177177
return true;
+160
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,160 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace SPC\command;
6+
7+
use SPC\store\FileSystem;
8+
use Symfony\Component\Console\Attribute\AsCommand;
9+
use Symfony\Component\Console\Input\InputArgument;
10+
use Symfony\Component\Console\Input\InputOption;
11+
12+
#[AsCommand(name: 'dump-extensions', description: 'Determines the required php extensions')]
13+
class DumpExtensionsCommand extends BaseCommand
14+
{
15+
protected bool $no_motd = true;
16+
17+
public function configure(): void
18+
{
19+
// path to project files or specific composer file
20+
$this->addArgument('path', InputArgument::OPTIONAL, 'Path to project root', '.');
21+
$this->addOption('format', 'F', InputOption::VALUE_REQUIRED, 'Parsed output format', 'default');
22+
// output zero extension replacement rather than exit as failure
23+
$this->addOption('no-ext-output', 'N', InputOption::VALUE_REQUIRED, 'When no extensions found, output default combination (comma separated)');
24+
// no dev
25+
$this->addOption('no-dev', null, null, 'Do not include dev dependencies');
26+
// no spc filter
27+
$this->addOption('no-spc-filter', 'S', null, 'Do not use SPC filter to determine the required extensions');
28+
}
29+
30+
public function handle(): int
31+
{
32+
$path = FileSystem::convertPath($this->getArgument('path'));
33+
34+
$path_installed = FileSystem::convertPath(rtrim($path, '/\\') . '/vendor/composer/installed.json');
35+
$path_lock = FileSystem::convertPath(rtrim($path, '/\\') . '/composer.lock');
36+
37+
$ext_installed = $this->extractFromInstalledJson($path_installed, !$this->getOption('no-dev'));
38+
if ($ext_installed === null) {
39+
if ($this->getOption('format') === 'default') {
40+
$this->output->writeln('<comment>vendor/composer/installed.json load failed, skipped</comment>');
41+
}
42+
$ext_installed = [];
43+
}
44+
45+
$ext_lock = $this->extractFromComposerLock($path_lock, !$this->getOption('no-dev'));
46+
if ($ext_lock === null) {
47+
$this->output->writeln('<error>composer.lock load failed</error>');
48+
return static::FAILURE;
49+
}
50+
51+
$extensions = array_unique(array_merge($ext_installed, $ext_lock));
52+
sort($extensions);
53+
54+
if (empty($extensions)) {
55+
if ($this->getOption('no-ext-output')) {
56+
$this->outputExtensions(explode(',', $this->getOption('no-ext-output')));
57+
return static::SUCCESS;
58+
}
59+
$this->output->writeln('<error>No extensions found</error>');
60+
return static::FAILURE;
61+
}
62+
63+
$this->outputExtensions($extensions);
64+
return static::SUCCESS;
65+
}
66+
67+
private function filterExtensions(array $requirements): array
68+
{
69+
return array_map(
70+
fn ($key) => substr($key, 4),
71+
array_keys(
72+
array_filter($requirements, function ($key) {
73+
return str_starts_with($key, 'ext-');
74+
}, ARRAY_FILTER_USE_KEY)
75+
)
76+
);
77+
}
78+
79+
private function loadJson(string $file): array|bool
80+
{
81+
if (!file_exists($file)) {
82+
return false;
83+
}
84+
85+
$data = json_decode(file_get_contents($file), true);
86+
if (!$data) {
87+
return false;
88+
}
89+
return $data;
90+
}
91+
92+
private function extractFromInstalledJson(string $file, bool $include_dev = true): ?array
93+
{
94+
if (!($data = $this->loadJson($file))) {
95+
return null;
96+
}
97+
98+
$packages = $data['packages'] ?? [];
99+
100+
if (!$include_dev) {
101+
$packages = array_filter($packages, fn ($package) => !in_array($package['name'], $data['dev-package-names'] ?? []));
102+
}
103+
104+
return array_merge(
105+
...array_map(fn ($x) => isset($x['require']) ? $this->filterExtensions($x['require']) : [], $packages)
106+
);
107+
}
108+
109+
private function extractFromComposerLock(string $file, bool $include_dev = true): ?array
110+
{
111+
if (!($data = $this->loadJson($file))) {
112+
return null;
113+
}
114+
115+
// get packages ext
116+
$packages = $data['packages'] ?? [];
117+
$exts = array_merge(
118+
...array_map(fn ($package) => $this->filterExtensions($package['require'] ?? []), $packages)
119+
);
120+
121+
// get dev packages ext
122+
if ($include_dev) {
123+
$packages = $data['packages-dev'] ?? [];
124+
$exts = array_merge(
125+
$exts,
126+
...array_map(fn ($package) => $this->filterExtensions($package['require'] ?? []), $packages)
127+
);
128+
}
129+
130+
// get require ext
131+
$platform = $data['platform'] ?? [];
132+
$exts = array_merge($exts, $this->filterExtensions($platform));
133+
134+
// get require-dev ext
135+
if ($include_dev) {
136+
$platform = $data['platform-dev'] ?? [];
137+
$exts = array_merge($exts, $this->filterExtensions($platform));
138+
}
139+
140+
return $exts;
141+
}
142+
143+
private function outputExtensions(array $extensions): void
144+
{
145+
if (!$this->getOption('no-spc-filter')) {
146+
$extensions = $this->parseExtensionList($extensions);
147+
}
148+
switch ($this->getOption('format')) {
149+
case 'json':
150+
$this->output->writeln(json_encode($extensions, JSON_PRETTY_PRINT));
151+
break;
152+
case 'text':
153+
$this->output->writeln(implode(',', $extensions));
154+
break;
155+
default:
156+
$this->output->writeln('<info>Required PHP extensions' . ($this->getOption('no-dev') ? ' (without dev)' : '') . ':</info>');
157+
$this->output->writeln(implode(',', $extensions));
158+
}
159+
}
160+
}

0 commit comments

Comments
 (0)