项目结构简介
static-php-cli 主要包含三种逻辑组件:资源、依赖库、扩展。这三种组件四个配置文件:source.json
、lib.json
、ext.json
、pkg.json
。
一个完整的构建静态 PHP 流程是:
- 使用资源下载模块
Downloader
下载指定或所有资源,这些资源包含 PHP 源码、依赖库源码、扩展源码。 - 使用资源解压模块
SourceExtractor
解压下载的资源到编译目录。 - 使用依赖工具计算出当前加入的扩展的依赖扩展、依赖库,然后对每个需要编译的依赖库进行编译,按照依赖顺序。
- 使用对应操作系统下的
Builder
构建每个依赖库后,将其安装到buildroot
目录。 - 如果包含外部扩展(源码没有包含在 PHP 内的扩展),将外部扩展拷贝到
source/php-src/ext/
目录。 - 使用
Builder
构建 PHP 源码,将其安装到buildroot
目录。
项目主要分为几个文件夹:
bin/
: 用于存放程序入口文件,包含bin/spc
、bin/spc-alpine-docker
、bin/setup-runtime
。config/
: 包含了所有项目支持的扩展、依赖库以及这些资源下载的地址、下载方式等,:lib.json
、ext.json
、source.json
、pkg.json
、pre-built.json
。src/SPC/
: 项目的核心代码,包含了整个框架以及编译各种扩展和库的命令。src/globals/
: 项目的全局方法和常量、运行时需要的测试文件(例如:扩展的可用性检查代码)。vendor/
: Composer 依赖的目录,你无需对它做出任何修改。
其中运行原理就是启动一个 symfony/console
的 ConsoleApplication
,然后解析用户在终端输入的命令。
基本命令行结构
bin/spc
是一个 PHP 代码入口文件,包含了 Unix 通用的 #!/usr/bin/env php
用来让系统自动以系统安装好的 PHP 解释器执行。 在项目执行了 new ConsoleApplication()
后,框架会自动使用反射的方式,解析 src/SPC/command
目录下的所有类,并将其注册成为命令。
项目并没有直接使用 Symfony 推荐的 Command 注册方式和命令执行方式,这里做出了一点小变动:
- 每个命令都使用
#[AsCommand()]
Attribute 来注册名称和简介。 - 将
execute()
抽象化,让所有命令基于BaseCommand
(它基于Symfony\Component\Console\Command\Command
),每个命令本身的执行代码写到了handle()
方法中。 BaseCommand
添加了变量$no_motd
,用于是否在该命令执行时显示 Figlet 欢迎词。BaseCommand
将InputInterface
和OutputInterface
保存为成员变量,你可以在命令的类内使用$this->input
和$this->output
。
基本源码结构
项目的源码位于 src/SPC
目录,支持 PSR-4 标准的自动加载,包含以下子目录和类:
src/SPC/builder/
: 用于不同操作系统下构建依赖库、PHP 及相关扩展的核心编译命令代码,还包含了一些编译的系统工具方法。src/SPC/command/
: 项目的所有命令都在这里。src/SPC/doctor/
: Doctor 模块,它是一个较为独立的用于检查系统环境的模块,可使用命令bin/spc doctor
进入。src/SPC/exception/
: 异常类。src/SPC/store/
: 有关存储、文件和资源的类都在这里。src/SPC/util/
: 一些可以复用的工具方法都在这里。src/SPC/ConsoleApplication.php
: 命令行程序入口文件。
如果你阅读过源码,你可能会发现还有一个 src/globals/
目录,它是用于存放一些全局变量、全局方法、构建过程中依赖的非 PSR-4 标准的 PHP 源码,例如测试扩展代码等。
Phar 应用目录问题
和其他 php-cli 项目一样,spc 自身对路径有额外的考虑。 因为 spc 可以在 php-cli directly
、micro SAPI
、php-cli with Phar
、vendor with Phar
等多种模式下运行,各类根目录存在歧义。这里会进行一个完整的说明。 此问题一般常见于 PHP 项目中存取文件的基类路径选择问题,尤其是在配合 micro.sfx
使用时容易出现路径问题。
注意,此处仅对你在开发 Phar 项目或 PHP 框架时可能有用。
接下来我们都将
static-php-cli
(也就是 spc)当作一个普通的php
命令行程序来看,你可以将 spc 理解为你自己的任何 php-cli 应用以参考。
下面主要有三个基本的常量理论值,我们建议你在编写 php 项目时引入这三种常量:
WORKING_DIR
:执行 PHP 脚本时的工作目录SOURCE_ROOT_DIR
或ROOT_DIR
:项目文件夹的根目录,一般为composer.json
所在目录FRAMEWORK_ROOT_DIR
:使用框架的根目录,自行开发的框架可能会用到,一般框架目录为只读
你可以在你的框架或者 cli 应用程序入口中定义这些常量,以方便在你的项目中使用路径。
下面是 PHP 内置的常量值,在 PHP 解释器内部已被定义:
__DIR__
:当前执行脚本的文件所在目录__FILE__
:当前执行脚本的文件路径
Git 项目模式(source)
Git 项目模式指的是一个框架或程序本身在当前文件夹以纯文本形式存放,运行通过 php path/to/entry.php
方式。
假设你的项目存放在 /home/example/static-php-cli/
目录下,或你的项目就是框架本身,里面包含 composer.json
等项目文件:
composer.json
src/App/MyCommand.app
vendor/*
bin/entry.php
我们假设从 src/App/MyCommand.php
中获取以上常量:
Constant | Value |
---|---|
WORKING_DIR | /home/example/static-php-cli |
SOURCE_ROOT_DIR | /home/example/static-php-cli |
FRAMEWORK_ROOT_DIR | /home/example/static-php-cli |
__DIR__ | /home/example/static-php-cli/src/App |
__FILE__ | /home/example/static-php-cli/src/App/MyCommand.php |
这种情况下,WORKING_DIR
、SOURCE_ROOT_DIR
、FRAMEWORK_ROOT_DIR
的值是完全一致的:/home/example/static-php-cli
。 框架的源码和应用的源码都在当前路径下。
Vendor 库模式(vendor)
Vendor 库模式一般是指你的项目为框架类或者被其他应用作为 composer 依赖项安装到项目中,存放位置在 vendor/author/XXX
目录。
假设你的项目是 crazywhalecc/static-php-cli
,你或其他人在另一个项目使用 composer require
安装了这个项目。
我们假设 static-php-cli 中包含同 Git 模式
的除 vendor
目录外的所有文件,并从 src/App/MyCommand
中获取常量值, 目录常量应该是:
Constant | Value |
---|---|
WORKING_DIR | /home/example/another-app |
SOURCE_ROOT_DIR | /home/example/another-app |
FRAMEWORK_ROOT_DIR | /home/example/another-app/vendor/crazywhalecc/static-php-cli |
__DIR__ | /home/example/another-app/vendor/crazywhalecc/static-php-cli/src/App |
__FILE__ | /home/example/another-app/vendor/crazywhalecc/static-php-cli/src/App/MyCommand.php |
这里的 SOURCE_ROOT_DIR
就指的是使用 static-php-cli
的项目的根目录。
Git 项目 Phar 模式(source-phar)
Git 项目 Phar 模式指的是将 Git 项目模式的项目目录打包为一个 phar
文件的模式。我们假设 /home/example/static-php-cli
将打包为一个 Phar 文件,目录有以下文件:
composer.json
src/App/MyCommand.app
vendor/*
bin/entry.php
打包为 app.phar
并存放到 /home/example/static-php-cli
目录下时,此时执行 app.phar
,假设执行了 src/App/MyCommand
代码,常量在该文件内获取:
Constant | Value |
---|---|
WORKING_DIR | /home/example/static-php-cli |
SOURCE_ROOT_DIR | phar:///home/example/static-php-cli/app.phar/ |
FRAMEWORK_ROOT_DIR | phar:///home/example/static-php-cli/app.phar/ |
__DIR__ | phar:///home/example/static-php-cli/app.phar/src/App |
__FILE__ | phar:///home/example/static-php-cli/app.phar/src/App/MyCommand.php |
因为在 phar 内读取自身 phar 的文件需要 phar://
协议进行,所以项目根目录和框架目录将会和 WORKING_DIR
不同。
Vendor 库 Phar 模式(vendor-phar)
Vendor 库 Phar 模式指的是你的项目作为框架安装在其他项目内,存储于 vendor
目录下。
我们假设你的项目目录结构如下:
composer.json # 当前项目的 Composer 配置文件
box.json # 打包 Phar 的配置文件
another-app.php # 另一个项目的入口文件
vendor/crazywhalecc/static-php-cli/* # 你的项目被作为依赖库
将该目录 /home/example/another-app/
下的这些文件打包为 app.phar
时,对于你的项目而言,下面常量的值应为:
Constant | Value |
---|---|
WORKING_DIR | /home/example/another-app |
SOURCE_ROOT_DIR | phar:///home/example/another-app/app.phar/ |
FRAMEWORK_ROOT_DIR | phar:///home/example/another-app/app.phar/vendor/crazywhalecc/static-php-cli |
__DIR__ | phar:///home/example/another-app/app.phar/vendor/crazywhalecc/static-php-cli/src/App |
__FILE__ | phar:///home/example/another-app/app.phar/vendor/crazywhalecc/static-php-cli/src/App/MyCommand.php |