检测 PHP 脚本是否正在交互运行


Detect if a PHP script is being run interactively or not

我有一个脚本,旨在既可以作为网页运行,也可以通过控制台运行。

检测使用哪种方法来调用脚本似乎非常简单,但是当脚本从控制台运行时,我需要知道脚本是否以交互方式运行(用户键入命令或从文件重定向的输入)。

php script.phpphp script.php < input_file

这可能吗?

我还需要一个比posix_isatty稍微灵活的解决方案来检测:

  • 脚本是否从终端运行
  • 脚本是通过管道还是从文件接收数据
  • 输出是否被重定向到文件

经过一些实验和挖掘libc标头,我想出了一个非常简单的类,可以完成上述所有操作以及更多操作。

class IOMode
{
    public $stdin;
    public $stdout;
    public $stderr;
    private function getMode(&$dev, $fp)
    {
        $stat = fstat($fp);
        $mode = $stat['mode'] & 0170000; // S_IFMT
        $dev = new StdClass;
        $dev->isFifo = $mode == 0010000; // S_IFIFO
        $dev->isChr  = $mode == 0020000; // S_IFCHR
        $dev->isDir  = $mode == 0040000; // S_IFDIR
        $dev->isBlk  = $mode == 0060000; // S_IFBLK
        $dev->isReg  = $mode == 0100000; // S_IFREG
        $dev->isLnk  = $mode == 0120000; // S_IFLNK
        $dev->isSock = $mode == 0140000; // S_IFSOCK
    }
    public function __construct()
    {
        $this->getMode($this->stdin,  STDIN);
        $this->getMode($this->stdout, STDOUT);
        $this->getMode($this->stderr, STDERR);
    }
}
$io = new IOMode;

一些示例用法,以显示它可以检测到的内容。

输入:

$ php io.php
// Character device as input
// $io->stdin->isChr  == true
$ echo | php io.php
// Input piped from another command
// $io->stdin->isFifo == true
$ php io.php < infile
// Input from a regular file (name taken verbatim from C headers)
// $io->stdin->isReg  == true
$ mkdir test
$ php io.php < test
// Directory used as input
// $io->stdin->isDir  == true

输出:

$ php io.php
// $io->stdout->isChr  == true
$ php io.php | cat
// $io->stdout->isFifo == true
$ php io.php > outfile
// $io->stdout->isReg  == true

错误:

$ php io.php
// $io->stderr->isChr  == true
$ php io.php 2>&1 | cat
// stderr redirected to stdout AND piped to another command
// $io->stderr->isFifo == true
$ php io.php 2>error
// $io->stderr->isReg  == true

我没有包括链接、套接字或块设备的示例,但没有理由它们不应该工作,因为它们的设备模式掩码在类中。

(未在 Windows 上测试 - 里程可能会有所不同)

posix_isatty()

if (posix_isatty(0)) {
  // STDIN is a TTY
} else {
  // STDIN is a pipe or has no associated TTY
}

显然,这仅适用于符合POSIX的操作系统,其中PHP安装了posix扩展。我不知道温多兹的等价物。

对于 Windows,您可以使用以下内容:

$b = stream_isatty(STDIN);
if ($b) {
   echo "php script.php'n";
} else {
   echo "php script.php < input_file'n";
}

https://php.net/function.stream-isatty