PHP 读取 STDIN 时出现语法错误
PHP Syntax Error when reading STDIN
我正在 PHP 中构建一个 CLI 脚本,它需要从 STDIN 读取内容并将其处理成一个变量。它是可执行的,但结果与调用 "php" 命令的参数相同。
这是一个由 WHM 事件挂钩触发运行的脚本。
这是文件:
#!/usr/bin/php -q
<?php
$switches = (count($argv) > 1) ? $argv : array();
if (in_array('--describe', $switches)) {
echo json_encode(describe());
exit;
} elseif (in_array('--debug', $switches)) {
list($status, $msg) = debug();
echo "$status $msg";
exit;
} else {
echo '0 myapp/whm.php needs a valid switch';
exit(1);
}
function getStdInStream()
{
$input = fopen('php://stdin', 'r');
$rawData = '';
if (is_resource($input)) {
stream_set_blocking($input, 0);
while (($line = fgets($input, 1024)) !== false) {
$rawData .= trim($line);
}
fclose($input);
}
if ($rawData) {
$data = json_decode($rawData, true);
} else {
$data = array('context'=>array(), 'data'=>array(), 'hook'=>array());
}
return $data;
}
function describe()
{
$debug = array(
'category' => 'Cpanel',
'event' => 'UAPI::Branding::get_application_information',
'stage' => 'pre',
'hook' => '/usr/local/cpanel/3rdparty/bin/hooktest --debug',
'exectype' => 'script',
);
return array($debug);
}
function debug()
{
$status = 1;
$msg = 'Triggered!';
$input = getStdInStream();
return array($status, $msg);
}
运行 命令行上的这个 returns 第 24 行的语法错误消息:
$ /usr/local/cpanel/3rdparty/bin/hooktest
<br />
<b>Parse error</b>: syntax error, unexpected '{' in <b>/usr/local/cpanel/3rdparty/bin/hooktest</b> on line <b>24</b><br />
我看不出语法有什么问题。我做了一些测试,比如在函数 getStdInStream
中注释代码,错误消失了。
该脚本基于此处找到的示例:
https://documentation.cpanel.net/display/SDK/Guide+to+Standardized+Hooks+-+Script+Hook+Action+Code
尝试逐行注释和取消注释,但在执行 fopen
后没有任何效果。
您是否知道为什么我无法使用该方法读取 STDIN,或者这是一个 PHP 错误?
我的PHP版本:
PHP 5.4.41 (cgi-fcgi) (built: May 22 2015 16:12:44)
Copyright (c) 1997-2014 The PHP Group
Zend Engine v2.4.0, Copyright (c) 1998-2014 Zend Technologies
刚刚找到似乎是解决方案的方法。
将 hashbang 从 #!/usr/bin/php -q
更改为 #!/usr/bin/env php
使得 PHP 对脚本的解释有所不同。
我注意到使用原始脚本(来自问题)未定义全局常量 STDIN
和 STDOUT
。使用 #!/usr/bin/env php
也可以解决该问题。
根据 env
的手册页,它在修改后的环境中运行程序,而不是 "just" 将脚本传递给解释器(在本例中为 php)。是什么让我认为它正确地将 stdin
和 stdout
重定向到环境中的脚本代码。
我正在 PHP 中构建一个 CLI 脚本,它需要从 STDIN 读取内容并将其处理成一个变量。它是可执行的,但结果与调用 "php" 命令的参数相同。
这是一个由 WHM 事件挂钩触发运行的脚本。
这是文件:
#!/usr/bin/php -q
<?php
$switches = (count($argv) > 1) ? $argv : array();
if (in_array('--describe', $switches)) {
echo json_encode(describe());
exit;
} elseif (in_array('--debug', $switches)) {
list($status, $msg) = debug();
echo "$status $msg";
exit;
} else {
echo '0 myapp/whm.php needs a valid switch';
exit(1);
}
function getStdInStream()
{
$input = fopen('php://stdin', 'r');
$rawData = '';
if (is_resource($input)) {
stream_set_blocking($input, 0);
while (($line = fgets($input, 1024)) !== false) {
$rawData .= trim($line);
}
fclose($input);
}
if ($rawData) {
$data = json_decode($rawData, true);
} else {
$data = array('context'=>array(), 'data'=>array(), 'hook'=>array());
}
return $data;
}
function describe()
{
$debug = array(
'category' => 'Cpanel',
'event' => 'UAPI::Branding::get_application_information',
'stage' => 'pre',
'hook' => '/usr/local/cpanel/3rdparty/bin/hooktest --debug',
'exectype' => 'script',
);
return array($debug);
}
function debug()
{
$status = 1;
$msg = 'Triggered!';
$input = getStdInStream();
return array($status, $msg);
}
运行 命令行上的这个 returns 第 24 行的语法错误消息:
$ /usr/local/cpanel/3rdparty/bin/hooktest
<br />
<b>Parse error</b>: syntax error, unexpected '{' in <b>/usr/local/cpanel/3rdparty/bin/hooktest</b> on line <b>24</b><br />
我看不出语法有什么问题。我做了一些测试,比如在函数 getStdInStream
中注释代码,错误消失了。
该脚本基于此处找到的示例: https://documentation.cpanel.net/display/SDK/Guide+to+Standardized+Hooks+-+Script+Hook+Action+Code
尝试逐行注释和取消注释,但在执行 fopen
后没有任何效果。
您是否知道为什么我无法使用该方法读取 STDIN,或者这是一个 PHP 错误?
我的PHP版本:
PHP 5.4.41 (cgi-fcgi) (built: May 22 2015 16:12:44)
Copyright (c) 1997-2014 The PHP Group
Zend Engine v2.4.0, Copyright (c) 1998-2014 Zend Technologies
刚刚找到似乎是解决方案的方法。
将 hashbang 从 #!/usr/bin/php -q
更改为 #!/usr/bin/env php
使得 PHP 对脚本的解释有所不同。
我注意到使用原始脚本(来自问题)未定义全局常量 STDIN
和 STDOUT
。使用 #!/usr/bin/env php
也可以解决该问题。
根据 env
的手册页,它在修改后的环境中运行程序,而不是 "just" 将脚本传递给解释器(在本例中为 php)。是什么让我认为它正确地将 stdin
和 stdout
重定向到环境中的脚本代码。