Как передать массив PHP скрипту BASH?
у меня есть php-скрипт и скрипт bash. Они в одном каталоге. Я запускаю PHP скрипт из командной строки, которая позволит передавать массив в скрипт bash. Я пытаюсь сделать следующее:
- передайте массив PHP скрипту BASH
- получить пользовательский ввод от STDIN
- передайте пользовательский ввод обратно в скрипт PHP для дальнейшей обработки
вот мой PHP-скрипт:
<?php
$a=array("red","green","blue","yellow");
$string = '(' . implode(' ', $a) . ')'; // (red green blue yellow)
$user_response = shell_exec('./response.sh $string');
// do something with $user_response
?>
предполагается скрипт BASH чтобы прочитать массив из STDIN и предложить пользователю выбрать опцию:
#!/bin/bash
options=$(); # (red green blue yellow) but this isn't working
i=0;
echo "select an option";
for each in "${options[@]}"
do
echo "[$i] $each"
i=$((i+1))
done
echo;
read input;
echo "You picked option ${options[$input]}";
# here's where I want to pass or export the input back to the
# php script for further processing
когда я запускаю PHP-скрипт, он не отображает варианты выбора.
4 ответов
вы можете иметь свой сценарий оболочки следующим образом:
#!/bin/bash
options=("$@")
i=0
echo "select an option"
for str in "${options[@]}"; do
echo "[$i] $str"
((i++))
done
echo
read -p 'Enter an option: ' input
echo "You picked option ${options[$input]}"
тогда ваш PHP-код выглядит так:
<?php
$a=array("red","green","blue","yellow");
$string = implode(' ', $a);
$user_response = shell_exec("./response.sh $string");
echo "$user_response\n";
?>
Однако имейте в виду, что вывод будет таким при запуске с PHP:
php -f test.php
Enter an option: 2
select an option
[0] red
[1] green
[2] blue
[3] yellow
You picked option blue
т. е. пользовательский ввод будет отображаться до вывода из скрипта.
Я бы сказал, что проще всего было бы не пытаться эмулировать внутренний массив bash, а использовать "нормальную" логику / пост-обработку. Например; если вы просто проходите implode(' ', $a)
к скрипту bash (вы также должны передать его через escapeshellarg()
):
$a=array("red","green","blue","yellow");
$args = implode(' ', array_map('escapeshellarg', $a));
$user_response = shell_exec('./response.sh '. $args);
затем вы можете пройти аргументы в bash с помощью
for each in $*; do
echo $each
done
проблема с вашим решением заключается в том, что вывод сценария оболочки фактически находится в PHP $response
переменной:
shell-скрипт:
#!/bin/bash
echo "Before prompt"
read -p 'Enter a value: ' input
echo "You entered $input"
PHP-скрипт:
<?php
$shell = shell_exec("./t.sh");
echo "SHELL RESPONSE\n$shell\n";
результат php t.php
:
$ php t.php
Enter a value: foo
SHELL RESPONSE
Before prompt
You entered foo
вы захватили весь STDOUT
сценария оболочки.
если вы хотите просто передать значения в скрипт, вариант $option_string = implode(' ', $array_of_values);
будет работать, чтобы разместить параметры индивидуально для сценария. Если бы ты ... как что - то немного более продвинутое (установка флагов, назначение вещей и т. д.) Попробуйте это (https://ideone.com/oetqaY):
function build_shell_args(Array $options = array(), $equals="=") {
static $ok_chars = '/^[-0-9a-z_:\/\.]+$/i';
$args = array();
foreach ($options as $key => $val) if (!is_null($val) && $val !== FALSE) {
$arg = '';
$key_len = 0;
if(is_string($key) && ($key_len = strlen($key)) > 0) {
if(!preg_match($ok_chars, $key))
$key = escapeshellarg($key);
$arg .= '-'.(($key_len > 1) ? '-' : '').$key;
}
if($val !== TRUE) {
if((string) $val !== (string) (int) $val) {
$val = print_r($val, TRUE);
if(!preg_match($ok_chars, $val))
$val = escapeshellarg($val);
}
if($key_len != 0)
$arg .= $equals;
$arg .= $val;
}
if(!empty($arg))
$args[] = $arg;
}
return implode(' ', $args);
}
это будет ваше самое полное решение для перехода в командную строку.
если вы вместо этого ищете способ запросить пользователя (в целом), я бы рассмотрел возможность остаться внутри PHP. Самый простой способ:
print_r("$question : ");
$fp = fopen('php://stdin', 'r');
$response = fgets($fp, 1024);
или, чтобы поддержать проверку вопроса, многострочный и только вызов Командной строки:
function prompt($message = NULL, $validator = NULL, $terminator = NULL, $include_terminating_line = FALSE) {
if(PHP_SAPI != 'cli') {
throw new \Exception('Can not Prompt. Not Interactive.');
}
$return = '';
// Defaults to 0 or more of any character.
$validator = !is_null($validator) ? $validator : '/^.*$/';
// Defaults to a lonely new-line character.
$terminator = !is_null($terminator) ? $terminator : "/^\n$/";
if(@preg_match($validator, NULL) === FALSE) {
throw new Exception("Prompt Validator Regex INVALID. - '$validator'");
}
if(@preg_match($terminator, NULL) === FALSE) {
throw new Exception("Prompt Terminator Regex INVALID. - '$terminator'");
}
$fp = fopen('php://stdin', 'r');
$message = print_r($message,true);
while (TRUE) {
print_r("$message : ");
while (TRUE) {
$line = fgets($fp, 1024); // read the special file to get the user input from keyboard
$terminate = preg_match($terminator, $line);
$valid = preg_match($validator, $line);
if (!empty($valid) && (empty($terminate) || $include_terminating_line)) {
$return .= $line;
}
if (!empty($terminate)) {
break 2;
}
if(empty($valid)) {
print_r("\nInput Invalid!\n");
break;
}
}
}
return $return;
}
поскольку скобки запускают то, что в них находится в под-оболочке, что не то, что я думаю, что вы хотите...
Я бы изменил это...
$string = '(' . implode(' ', $a) . ')';
для этого...
$string = '"' . implode (' ', $a) . '"';
кроме того, используйте двойные кавычки здесь...
$user_response = shell_exec ("./response.sh $string");
или вырваться...
$user_response = shell_exec ('./response.sh ' . $string);
поэтому я бы также изменил BASH, чтобы просто принять один аргумент, строку и разделить этот аргумент на массив, чтобы получить наши варианты.
вот так...
#!/bin/bash
IFS=' ';
read -ra options <<< "";
i=0;
echo "select an option";
for each in "${options[@]}"; do
echo "[$i] $each";
i=$((i+1));
done;
unset i;
echo;
read input;
echo "You picked option " ${options[$input]};