# Package 模型 ## Package 定义 Package 是 StaticPHP 构建系统中的核心概念,代表一个可构建/可安装的单元,如 PHP 扩展、库、构建目标等。 每个 Package 包含构建信息、依赖关系、构建逻辑等,构成了 StaticPHP 的构建模型。Package 的定义主要通过 YAML/JSON 配置文件来实现。`core` 注册表的包配置文件位于 `config/pkg/` 目录下,对应的构建类位于 `src/Package/` 目录下。 Package 主要分为四种类型: - **php-extension**:PHP 扩展包,包含 PHP 扩展的构建信息和构建逻辑。 - **library**:库包,包含构建工具链、依赖库等的构建信息和构建逻辑。 - **target**:构建目标包,代表最终的构建产物,如 PHP 二进制、curl 二进制等,继承自 `library` 包类型。 - **virtual-target**:虚构建目标包,代表一个抽象的构建目标,不直接对应构建产物,主要用于依赖管理和构建调度。 ```yaml {pkg-name}: type: {pkg-type} ... ``` ## Artifact 定义 Artifact 是独立于 Package 的定义,它包含构建包的源码归档文件或预构建的二进制文件。每个 Artifact 定义了下载 URL、解压方式、构建产物的文件路径等信息。Package 可以通过 `artifact` 字段引用一个或多个 Artifact 来获取构建所需的源码或二进制文件。 简单来说,默认情况下,一个 Package 对应一个 Artifact;如果多个 Package 共用一份源码时,可以定义一个 Artifact 供多个 Package 引用。Artifact 的定义位于 `config/artifact/` 目录下,对应的自定义下载/解压逻辑类位于 `src/Package/Artifact/` 目录下;对于虚拟目标、PHP 内置扩展等特殊包类型,Package 也可以不设置 Artifact 字段。 我们假设 `example-library-package` 是一个依赖库,它的源码归档文件托管在 `https://example.com/example-library.tar.gz`,则它的 Package 定义和 Artifact 定义可以如下所示: ```yaml example-library-package: type: library artifact: source: type: url url: 'https://example.com/example-library.tar.gz' ``` 更多有关 Artifact 定义的内容,请参阅 [Artifact 模型](./artifact-model) 章节。 ## php-extension 包类型 php-extension 一个包代表一个 PHP 扩展,它的配置文件位于 `config/pkg/ext/` 目录下,构建类继承自 `PhpExtensionPackage`,位于 `src/Package/Extension/` 目录下。PHP 扩展包的配置文件包含扩展名称、版本、依赖关系、构建选项等信息。 ```yaml ext-lz4: type: php-extension artifact: source: type: git url: 'https://github.com/kjdev/php-ext-lz4.git' rev: master extract: php-src/ext/lz4 metadata: license-files: [LICENSE] license: MIT depends: - liblz4 php-extension: arg-type@unix: '--enable-lz4=@shared_suffix@ --with-lz4-includedir=@build_root_path@' arg-type@windows: '--enable-lz4' ``` `php-extension` 允许的定义字段: ```yaml ext-{ext-name}: # 包名必须以 ext- 前缀开头 type: php-extension # ── 通用字段 ───────────────────────────────────────────────────────────── description: '...' # 可选,人类可读的包描述 lang: c # 可选,扩展的实现语言(c / c++ 等) frameworks: [] # 可选,相关macOS框架依赖列表 artifact: '{artifact-name}' # 可选;字符串时引用同名 Artifact 定义, # 对象时为内联 Artifact(内置扩展无需此字段) # depends / suggests 支持 @windows / @unix / @linux / @macos 后缀 depends: [] # 可选,硬依赖列表(库名直接写,PHP 扩展需加 ext- 前缀) depends@unix: [] # 可选,仅 Unix 平台生效的硬依赖 depends@windows: [] # 可选,仅 Windows 平台生效的硬依赖 suggests: [] # 可选,可选依赖列表(格式同 depends) suggests@unix: [] # ── php-extension 专属字段(嵌套在 php-extension: 对象中)───────────────── php-extension: # arg-type 决定传递给 ./configure 的参数形式,支持平台后缀 # 支持的平台后缀:@unix(Linux + macOS)、@linux、@macos、@windows # 优先级(以 Linux 为例):arg-type@linux > arg-type@unix > arg-type(无后缀) # 内置关键字: # enable → --enable-{extname}(默认值,未配置时使用) # enable-path → --enable-{extname}={buildroot} # with → --with-{extname} # with-path → --with-{extname}={buildroot} # custom/none → 不传递任何参数(由 PHP 类的 #[CustomPhpConfigureArg] 方法处理) # 也可直接写完整参数字符串,支持以下占位符: # @build_root_path@ → BUILD_ROOT_PATH(buildroot 绝对路径) # @shared_suffix@ → 共享构建时展开为 =shared,静态构建时为空 # @shared_path_suffix@ → 共享构建时展开为 =shared,{buildroot},静态构建时为 ={buildroot} arg-type: enable arg-type@unix: '--enable-{extname}=@shared_suffix@' arg-type@windows: with-path zend-extension: false # 可选,true 表示这是 Zend 扩展(如 opcache、xdebug) build-shared: true # 可选,是否允许构建为共享扩展(.so),默认 true build-static: true # 可选,是否允许内联静态构建(编译进 PHP),默认 true build-with-php: true # 可选,true 表示该扩展通过 PHP 源码树一同编译(内置扩展使用) # display-name 影响 smoke test 中 php --ri 的参数及许可证导出显示名称 # 不填时默认使用扩展名(ext- 后缀部分);填空字符串则跳过 --ri 检查 display-name: 'My Extension' # os 限制该扩展仅在指定平台上可用,不在列表内的平台会拒绝构建 # 可选值:Linux、Darwin、Windows os: [Linux, Darwin] ``` ## library 包类型 library 包代表一个需要从源码编译的依赖库(如 openssl、zlib 等),其配置文件位于 `config/pkg/lib/` 目录下,构建类继承自 `LibraryPackage`,位于 `src/Package/Library/` 目录下。 以 openssl 为例: ```yaml openssl: type: library artifact: source: type: ghrel repo: openssl/openssl match: openssl.+\.tar\.gz prefer-stable: true binary: hosted metadata: license-files: [LICENSE.txt] license: OpenSSL depends: - zlib depends@windows: - zlib - jom headers: - openssl static-libs@unix: - libssl.a - libcrypto.a static-libs@windows: - libssl.lib - libcrypto.lib ``` `library` 允许的定义字段: ```yaml {lib-name}: type: library # library 或 target(target 继承 library 的所有字段) # ── 通用字段 ───────────────────────────────────────────────────────────── description: '...' # 可选,人类可读的包描述 license: MIT # 可选,SPDX 许可证标识符(用于许可证导出) lang: c # 可选,库的实现语言(c / c++ 等) frameworks: [] # 可选,相关框架标签列表 artifact: '{artifact-name}' # 必填;字符串时引用同名 Artifact 定义,对象时为内联 Artifact # depends / suggests 支持 @windows / @unix / @linux / @macos 后缀 depends: [] # 可选,硬依赖列表(库名或 ext- 前缀的 PHP 扩展名) depends@unix: [] depends@windows: [] suggests: [] # 可选,可选依赖列表(格式同 depends) # ── library / target 专属字段 ──────────────────────────────────────────── # 以下字段用于构建完成后验证产物是否已正确安装,支持 @unix / @windows / @linux / @macos 后缀 # 验证 buildroot/include/ 下是否存在指定头文件或目录 # 相对路径基于 buildroot/include/,绝对路径直接使用 headers: - openssl # 对应 buildroot/include/openssl/ - zlib.h # 对应 buildroot/include/zlib.h headers@unix: - ffi.h # 验证 buildroot/lib/ 下是否存在指定静态库文件 # 相对路径基于 buildroot/lib/,绝对路径直接使用 static-libs@unix: - libssl.a static-libs@windows: - libssl.lib # 验证 buildroot/lib/pkgconfig/ 下是否存在指定 .pc 文件 # 仅在非 Windows 平台检查(pkg-config 在 Windows 上不适用) pkg-configs: - openssl # 对应 buildroot/lib/pkgconfig/openssl.pc - libssl # 自动补全 .pc 后缀 # 验证 buildroot/bin/ 下是否存在指定可执行文件 # 相对路径基于 buildroot/bin/,绝对路径直接使用 static-bins: - my-tool # 包安装完成后注入到全局 PATH 的目录列表,支持路径占位符(见下方说明) path: - '{pkg_root_path}/rust/bin' # 包安装完成后设置的环境变量(覆盖已有值),支持路径占位符 env: MY_VAR: '{build_root_path}/lib' # 包安装完成后追加到已有环境变量末尾的值,支持路径占位符 append-env: CFLAGS: ' -I{build_root_path}/include' ``` `path`、`env`、`append-env` 字段的字符串值中支持以下路径占位符: | 占位符 | 实际路径 | |---|---| | `{build_root_path}` | buildroot 目录(`buildroot/`) | | `{pkg_root_path}` | pkgroot 目录(`pkgroot/`) | | `{working_dir}` | 工作目录(项目根目录) | | `{download_path}` | 下载缓存目录(`downloads/`) | | `{source_path}` | 解压源码目录(`source/`) | | `{php_sdk_path}` | Windows PHP SDK 目录 | ## target 包类型 `target` 包代表一个最终的构建产物,它继承于 `library`,所以包含 `library` 的所有定义字段。`target` 包的配置文件位于 `config/pkg/target/` 目录下,构建类继承自 `TargetPackage`,位于 `src/Package/Target/` 目录下。 与 `library` 的唯一区别是,`target` 包可以注册成为构建目标,且自动注册构建命令 `spc build:{target-name}`。 ## virtual-target 包类型 与 `target` 不同的是,`virtual-target` 可以不包含 `artifact`,即不直接对应一个可构建的实体,而是一个抽象的构建目标,主要用于依赖管理和构建调度。`virtual-target` 的配置文件位于 `config/pkg/target/` 目录下,构建类继承自 `TargetPackage`,位于 `src/Package/Target/` 目录下。它的定义与 `target` 基本相同,但 `artifact` 字段可选且通常不设置。`virtual-target` 主要用于以下场景: - 定义一个抽象的构建目标,供其他包依赖,但不直接对应一个可构建的实体。 - 作为多个 `target` 包的公共依赖,简化依赖关系管理。 典型例子就是 PHP 包的 `php-cli`、`php-fpm` 等构建目标,他们没有独立的源码,依赖于 `php-src`,通过构建调度来决定最终构建成 CLI 还是 FPM 二进制。