diff --git a/src/ZM/Command/BuildCommand.php b/src/ZM/Command/BuildCommand.php
index 4728171c..b3cb65f3 100644
--- a/src/ZM/Command/BuildCommand.php
+++ b/src/ZM/Command/BuildCommand.php
@@ -13,6 +13,8 @@ use Symfony\Component\Console\Output\OutputInterface;
#[AsCommand(name: 'build', description: '将项目构建一个phar包')]
class BuildCommand extends Command
{
+ use NonPharLoadModeOnly;
+
/**
* 配置
*/
diff --git a/src/ZM/Command/CheckConfigCommand.php b/src/ZM/Command/CheckConfigCommand.php
index 5534e60b..651e4ef0 100644
--- a/src/ZM/Command/CheckConfigCommand.php
+++ b/src/ZM/Command/CheckConfigCommand.php
@@ -5,57 +5,52 @@ declare(strict_types=1);
namespace ZM\Command;
use Symfony\Component\Console\Attribute\AsCommand;
-use Symfony\Component\Console\Command\Command;
-use Symfony\Component\Console\Input\InputInterface;
-use Symfony\Component\Console\Output\OutputInterface;
#[AsCommand(name: 'check:config', description: '检查配置文件是否和框架当前版本有更新')]
class CheckConfigCommand extends Command
{
+ use SourceLoadModeOnly;
+
private $need_update = false;
- protected function execute(InputInterface $input, OutputInterface $output): int
+ protected function handle(): int
{
- if (LOAD_MODE !== 1) {
- $output->writeln('仅限在Composer依赖模式中使用此命令!');
- return 1;
- }
$current_cfg = getcwd() . '/config/';
- $remote_cfg = include_once FRAMEWORK_ROOT_DIR . '/config/global_old.php';
+ $remote_cfg = include FRAMEWORK_ROOT_DIR . '/config/global_old.php';
if (file_exists($current_cfg . 'global.php')) {
- $this->check($remote_cfg, 'global.php', $output);
+ $this->check($remote_cfg, 'global.php');
}
if (file_exists($current_cfg . 'global.development.php')) {
- $this->check($remote_cfg, 'global.development.php', $output);
+ $this->check($remote_cfg, 'global.development.php');
}
if (file_exists($current_cfg . 'global.staging.php')) {
- $this->check($remote_cfg, 'global.staging.php', $output);
+ $this->check($remote_cfg, 'global.staging.php');
}
if (file_exists($current_cfg . 'global.production.php')) {
- $this->check($remote_cfg, 'global.production.php', $output);
+ $this->check($remote_cfg, 'global.production.php');
}
if ($this->need_update === true) {
- $output->writeln('有配置文件需要更新,详情见文档 `https://framework.zhamao.xin/update/config`');
+ $this->comment('有配置文件需要更新,详情见文档 `https://framework.zhamao.xin/update/config`');
} else {
- $output->writeln('配置文件暂无更新!');
+ $this->info('配置文件暂无更新!');
}
- return 0;
+ return self::SUCCESS;
}
/**
* @param mixed $remote
* @param mixed $local
*/
- private function check($remote, $local, OutputInterface $out)
+ private function check($remote, $local)
{
- $local_file = include_once WORKING_DIR . '/config/' . $local;
+ $local_file = include WORKING_DIR . '/config/' . $local;
if ($local_file === true) {
$local_file = config('global');
}
foreach ($remote as $k => $v) {
if (!isset($local_file[$k])) {
- $out->writeln('配置文件 ' . $local . " 需要更新!(当前配置文件缺少 `{$k}` 字段配置)");
+ $this->comment("配置文件 {$local} 需要更新!(当前配置文件缺少 `{$k}` 字段配置)");
$this->need_update = true;
}
}
diff --git a/src/ZM/Command/Command.php b/src/ZM/Command/Command.php
index 9ded2f91..97893708 100644
--- a/src/ZM/Command/Command.php
+++ b/src/ZM/Command/Command.php
@@ -32,7 +32,20 @@ abstract class Command extends \Symfony\Component\Console\Command\Command
{
$this->input = $input;
$this->output = $output;
- return $this->handle();
+ if ($this->shouldExecute()) {
+ return $this->handle();
+ }
+ return self::SUCCESS;
+ }
+
+ /**
+ * 是否应该执行
+ *
+ * @return bool 返回 true 以继续执行,返回 false 以中断执行
+ */
+ protected function shouldExecute(): bool
+ {
+ return true;
}
/**
diff --git a/src/ZM/Command/InitCommand.php b/src/ZM/Command/InitCommand.php
index b45997da..48c028e2 100644
--- a/src/ZM/Command/InitCommand.php
+++ b/src/ZM/Command/InitCommand.php
@@ -13,6 +13,8 @@ use ZM\Exception\InitException;
#[AsCommand(name: 'init', description: '初始化框架运行的基础文件')]
class InitCommand extends Command
{
+ use NonPharLoadModeOnly;
+
private string $base_path;
private bool $force = false;
diff --git a/src/ZM/Command/NonPharLoadModeOnly.php b/src/ZM/Command/NonPharLoadModeOnly.php
new file mode 100644
index 00000000..7d85fdd0
--- /dev/null
+++ b/src/ZM/Command/NonPharLoadModeOnly.php
@@ -0,0 +1,20 @@
+error('此命令只能在非 Phar 模式下使用!');
+ }
+ return false;
+ }
+}
diff --git a/src/ZM/Command/Server/ServerStartCommand.php b/src/ZM/Command/Server/ServerStartCommand.php
index c4374c6d..b204aa31 100644
--- a/src/ZM/Command/Server/ServerStartCommand.php
+++ b/src/ZM/Command/Server/ServerStartCommand.php
@@ -14,7 +14,7 @@ use ZM\Exception\ZMKnownException;
use ZM\Framework;
use ZM\Process\ProcessStateManager;
-#[AsCommand(name: 'server:start', description: '启动服务器', aliases: ['server'])]
+#[AsCommand(name: 'server', description: '启动服务器', aliases: ['server:start'])]
class ServerStartCommand extends ServerCommand
{
public static function exportOptionArray(): array
diff --git a/src/ZM/Command/SourceLoadModeOnly.php b/src/ZM/Command/SourceLoadModeOnly.php
new file mode 100644
index 00000000..e9d3b946
--- /dev/null
+++ b/src/ZM/Command/SourceLoadModeOnly.php
@@ -0,0 +1,20 @@
+error('此命令只能在源码模式下使用!');
+ }
+ return false;
+ }
+}
diff --git a/src/ZM/ConsoleApplication.php b/src/ZM/ConsoleApplication.php
index 7a0c5120..c181609a 100644
--- a/src/ZM/ConsoleApplication.php
+++ b/src/ZM/ConsoleApplication.php
@@ -4,22 +4,13 @@ declare(strict_types=1);
namespace ZM;
-use Phar;
use Symfony\Component\Console\Application;
+use Symfony\Component\Console\Attribute\AsCommand;
+use Symfony\Component\Console\CommandLoader\FactoryCommandLoader;
use Symfony\Component\Console\Input\InputInterface;
use Symfony\Component\Console\Output\OutputInterface;
-use ZM\Command\BotCraft\BotCraftCommand;
-use ZM\Command\BuildCommand;
-use ZM\Command\CheckConfigCommand;
-use ZM\Command\Generate\SystemdGenerateCommand;
-use ZM\Command\InitCommand;
-use ZM\Command\ProxyServerCommand;
-use ZM\Command\ReplCommand;
-use ZM\Command\Server\ServerReloadCommand;
-use ZM\Command\Server\ServerStartCommand;
-use ZM\Command\Server\ServerStatusCommand;
-use ZM\Command\Server\ServerStopCommand;
use ZM\Exception\SingletonViolationException;
+use ZM\Store\FileSystem;
/**
* 命令行启动的入口文件,用于初始化环境变量,并启动命令行应用
@@ -37,24 +28,29 @@ final class ConsoleApplication extends Application
}
// 初始化命令
- $this->add(new ServerStatusCommand()); // server运行状态
- $this->add(new ServerReloadCommand()); // server重载
- $this->add(new ServerStopCommand()); // server停止
- $this->add(new ServerStartCommand()); // 运行主服务的指令控制器
- $this->add(new SystemdGenerateCommand()); // 生成systemd文件
- $this->add(new BotCraftCommand()); // 用于从命令行创建插件
- $this->add(new ReplCommand()); // 交互式控制台
- $this->add(new ProxyServerCommand()); // HTTP 代理服务器
- if (LOAD_MODE === 1) { // 如果是 Composer 模式加载的,那么可以输入 check:config 命令,检查配置文件是否需要更新
- $this->add(new CheckConfigCommand());
- }
- if (\Phar::running() === '') { // 不是 Phar 模式的话,可以执行打包解包初始化命令
- $this->add(new BuildCommand()); // 用于将整个应用打包为一个可执行的 phar
- $this->add(new InitCommand()); // 用于在 Composer 模式启动下,初始化脚手架文件
- // $this->add(new PluginPackCommand()); // 用于打包一个子模块为 phar 并进行分发
- // $this->add(new PluginListCommand()); // 用于列出已配置的子模块列表(存在 zm.json 文件的目录)
- // $this->add(new PluginUnpackCommand()); // 用于将打包好的 phar 模块解包到 src 目录中
+ $command_classes = FileSystem::getClassesPsr4(zm_dir('src/ZM/Command'), 'ZM\\Command');
+ $commands = [];
+ foreach ($command_classes as $command_class) {
+ try {
+ $command_class_ref = new \ReflectionClass($command_class);
+ } catch (\ReflectionException $e) {
+ logger()->error("命令 {$command_class} 无法加载!反射失败:" . $e->getMessage());
+ continue;
+ }
+ if ($command_class_ref->isAbstract()) {
+ continue;
+ }
+ // 从 AsCommand 注解中获取命令名称
+ $attr = $command_class_ref->getAttributes(AsCommand::class);
+ if (count($attr) > 0) {
+ $commands[$attr[0]->getArguments()['name']] = fn () => new $command_class();
+ } else {
+ logger()->warning("命令 {$command_class} 没有使用 AsCommand 注解,无法被加载");
+ }
}
+ // 命令工厂,用于延迟加载命令
+ $command_loader = new FactoryCommandLoader($commands);
+ $this->setCommandLoader($command_loader);
self::$obj = $this; // 用于标记已经初始化完成
parent::__construct($name, ZM_VERSION);