在PHP中使用OS进程

2023-06-01 00:00:00 os php 进程

有时您需要使用 PHP 应用程序中的操作系统级命令。

让我们看看我们如何做到这一点,看看我们是否可以让开发者体验更好。


在过去的几年里,我一直专注于我如何编写代码以及如何改进它的各个方面。

我首先研究如何使与 HTTP 的集成更好、更面向对象。

我相信我找到了实现这一目标的方法,现在我将注意力集中在其他地方。


在某些情况下,您希望在应用程序中使用 OS CLI。在 Web 应用程序或另一个 CLI 应用程序中。

过去,我们使用过类似execorpassthru或shell_execand的方法system。然后出现了 Symfony Process 组件,我们得救了。


Symfony 进程组件使得与操作系统进程集成并获得输出变得非常容易。但是我们如何与这个库集成仍然有点令人沮丧。我们创建一个新进程,传入一个参数数组,使我们希望运行的命令。让我们来看看:

$command = new Process(
command: ['git', 'push', 'origin', 'main'],
);
 
$command->run();

这种方法有什么问题?

好吧,老实说,什么都没有。但是有没有办法改善开发人员的体验?

假设我们从 git 切换到 svn(我不太可能知道)。

为了改善开发人员的体验,首先,我们需要了解逻辑上用于创建 OS 命令的组件。

我们可以将它们分解为:

可执行
参数


我们的可执行文件是我们直接与之交互的东西,例如 php、git、brew 或我们系统上任何其他已安装的二进制文件。然后争论是我们如何互动;这些可以是子命令、选项、标志或参数。

因此,如果我们稍微抽象一下,我们将得到一个带参数的 aprocess和 a command。

我们将使用接口/契约来定义我们的组件来控制我们的工作流程应该如何工作。

让我们从流程契约开始:

declare(strict_types=1);
 
namespace JustSteveKing\OS\Contracts;
use Symfony\Component\Process\Process;
 
interface ProcessContract
{
   public function build(): Process;
}

我们这里是说每个进程都必须能够被构建,并且创建的进程的结果应该是一个 Symfony 进程。

我们的流程应该构建一个命令供我们运行,所以现在让我们看看我们的命令契约:

declare(strict_types=1);
 
namespace JustSteveKing\OS\Contracts;
 
interface CommandContract
{
   public function toArgs(): array;
}

我们希望从命令中得到的主要内容是能够作为参数返回,我们可以将这些参数作为命令传递给 Symfony 进程。想法已经够多了,让我们来看一个真实的例子。

我们将使用 git 作为示例,因为我们大多数人应该能够与 git 命令相关联。

首先,让我们创建一个 Git 进程来实现我们刚刚描述的 Process Contract:

class Git implements ProcessContract
{
   use HandlesGitCommands;
   private CommandContract $command;
}

我们的流程实现了合约,并有一个命令属性,我们将使用它允许我们的流程被流畅地构建和执行。

我们有一个特点,可以让我们集中精力为我们的 Git 流程构建和制造事物的方式。

让我们看一下:

trait HandlesGitCommands
{
   public function build(): Process
   {
       return new Process(
           command: $this->command->toArgs(),
       );
   }
 
   protected function buildCommand(Git $type, array $args = []): void
   {
       $this->command = new GitCommand(
           type: $type,
           args: $args,
       );
   }
}

因此,我们的 trait 展示了流程契约本身的实现,并提供了有关如何构建流程的说明。

它还包含一个允许我们抽象构建命令的方法。


到目前为止,我们可以创建一个流程并建立一个潜在的命令。

但是,我们还没有下达命令。我们在 trait 中创建一个新的 Git 命令,它使用 Git 类作为类型。

让我们看看另一个 Git 类,它是一个枚举。不过,我将展示一个精简版本 - 实际上,您希望它映射到您希望支持的所有 git 子命令:

enum Git: string
{
   case PUSH = 'push';
   case COMMIT = 'commit';
}

然后我们将它传递给 Git 命令:

final class GitCommand implements CommandContract
{
   public function __construct(
       public readonly Git $type,
       public readonly array $args = [],
       public readonly null|string $executable = null,
   )
   {
   }
 
   public function toArgs(): array
   {
       $executable = (new ExecutableFinder())->find(
           name: $this->executable ?? 'git',
       );
        
       if (null === $executable) {
           throw new InvalidArgumentException(
           message: "Cannot find executable for [$this->executable].",
       );
   }
 
       return array_merge(
           [$executable],
           [$this->type->value],
           $this->args,
       );
   }
}

在这个类中,我们接受来自我们的 Process 的参数,目前由我们的HandledGitCommandstrait 处理。

然后我们可以把它变成 Symfony 进程可以理解的参数。

我们使用ExecutableFinderSymfony 包中的包来最大程度地减少路径中的错误。

但是,如果找不到可执行文件,我们也想抛出异常。


当我们把它们放在我们的 Git 进程中时,它看起来有点像这样:

use JustSteveKing\OS\Commands\Types\Git as SubCommand;
 
class Git implements ProcessContract
{
   use HandlesGitCommands;
   private CommandContract $command;
 
   public function push(string $branch): Process
   {
       $this->buildCommand(
       type: SubCommand:PUSH,
       args: [
           'origin',
           $branch,
           ],
       );
    
       return $this->build();
   }
}

现在剩下要做的就是运行代码本身,以便我们可以在 PHP 应用程序中很好地使用 git:

$git = new Git();
$command = $git->push(
   branch: 'main',
);
 
$result = $command->run();

push 方法的结果将允许您与 Symfony 进程进行交互——这意味着您可以使用另一端的命令进行各种操作。我们唯一改变的是围绕这个过程的创建构建一个面向对象的包装器。这使我们能够很好地开发和保持上下文,并以可测试和可扩展的方式扩展事物。


您多久在应用程序中使用操作系统命令?

你能想到任何用例吗?我已经在 GitHub 上的 repo 中发布了示例代码,以便您可以使用它并查看是否可以改进您的操作系统集成。

https://github.com/JustSteveKing/os-process

一个很好的例子应该是 SSH、MySQL,甚至是 ansible 或 terraform!

想象一下,如果您可以在 Laravel artisan 的时间表上高效地运行 MySQL 转储,而无需一直使用第三方包!


转:

https://laravel-news.com/working-with-os-process-in-php

相关文章