Как передать массив PHP скрипту BASH?

у меня есть php-скрипт и скрипт bash. Они в одном каталоге. Я запускаю PHP скрипт из командной строки, которая позволит передавать массив в скрипт bash. Я пытаюсь сделать следующее:

  1. передайте массив PHP скрипту BASH
  2. получить пользовательский ввод от STDIN
  3. передайте пользовательский ввод обратно в скрипт 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]};