Как получить ошибки командной строки в файле журнала ошибок php?

Я пытаюсь бежать curl команда в файле php и пытается вывести ее вывод в файл журнала ошибок. Все ошибки php перечислены в файле кроме ошибок, которые я получаю от exec() функция. PHP-код, который я пытаюсь это:

exec("nohup curl --output ".$_SERVER['DOCUMENT_ROOT']."/abc.mp3 http://192.99.8.170:8098/stream/1; --max-time $time_in_seconds > /devnull&");

Может кто-нибудь, пожалуйста, помогите, чтобы я мог получить ошибку, сгенерированную этой командой в файле журнала ошибок. Я гуглил по этому поводу, но не смог получить достаточных результатов.

Спасибо!

1 ответ

Решение

Команда

В самой команде есть несколько проблем.

exec ("nohup curl --output". $ _ SERVER ['DOCUMENT_ROOT']. "/abc.mp3 http://192.99.8.170:8098/stream/1; --max-time $ time_in_seconds> / devnull &");

(Я обернул оригинальную строку для ясности.)

Нет необходимости в nohup

Основное назначение nohup Команда должна запустить процесс, если вы вышли из оболочки из-за зависания линии терминала. (Когда оболочка завершается, она отправляет SIGHUP своим дочерним процессам.) nohup Вы можете запустить команду, а затем выйти из оболочки.

curl команда действительно прекращена на SIGHUP. Вы можете использовать nohup поставить процесс загрузки в фоновом режиме. Но ваша цель - захватить вывод команды, будь то STDIN или STDOUT. Также важно знать, когда команда завершает свою работу и как. Например, после синхронного выполнения команды можно проверить код завершения (ноль указывает на успешность) и проверить, действительно ли файл загружен в файловую систему:

$output_dir = empty($_SERVER['DOCUMENT_ROOT']) ?
'/tmp/default-download-dir' : $_SERVER['DOCUMENT_ROOT'];
if (!is_dir($output_dir)) {
  if (!mkdir($output_dir, 0755, true))
    die("Failed to create directory $output_dir");
}

$url = 'http://php.net/images/logo.php';
$output_file = "{$output_dir}/logo.svg"; ;
// 2>&1 redirects standard error descriptor (2)
// to the standard output descriptor (1)
$command = sprintf("curl -o %s %s 2>&1",
  escapeshellarg($output_file),
  escapeshellarg($url));
$output = [];
exec($command, $output, $exit_code);
if ($exit_code != 0) {
  die("Failed to download $url to $output_file");
}

clearstatcache(true, $output_file);
if (!file_exists($output_file)) {
  die("File $output_file not found on file system, command: $command");
}

echo "$url has been saved to $output_file:\n",
  implode(PHP_EOL, $output), PHP_EOL;

Выход

http://php.net/images/logo.php has been saved to /tmp/default-download-dir/logo.svg:
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
0     0    0     0    0     0      0      0 --:--:-- --:--:-- --:--:--     0
100  2202  100  2202    0     0   4171    0 --:--:-- --:--:-- --:--:--  4665

Обратите внимание на использование escapeshellarg функция. Вы должны экранировать все аргументы оболочки, иначе оболочка будет интерпретировать свои специальные символы.

Точка с запятой

Строки разделяются точкой с запятой (;) интерпретируются как разные команды. Так что ваши --max-time $time_in_seconds > /devnull& интерпретируется как вторая команда. Я понятия не имею, чего вы пытались достичь, но, скорее всего, вам это не нужно.

Перенаправление на /dev/null

В твоей опечатке > /devnull& выражение. Вы, вероятно, имели в виду перенаправление на /dev/null псевдо-устройства: > /dev/null,

Захват вывода в разные файловые дескрипторы

В приведенном выше коде мы зафиксировали стандартный вывод. Вы должны использовать proc_open() функция, если вы хотите захватить вывод и для других файловых дескрипторов:

$output_dir = empty($_SERVER['DOCUMENT_ROOT']) ?
  '/tmp/default-download-dir' : $_SERVER['DOCUMENT_ROOT'];
if (!is_dir($output_dir)) {
  if (!mkdir($output_dir, 0755, true))
    die("Failed to create directory $output_dir");
}

$url = 'http://php.net/images/logo.php';
$output_file = "{$output_dir}/logo.svg"; ;
$command = sprintf("curl -o %s %s",
  escapeshellarg($output_file),
  escapeshellarg($url));

$descriptors = [
  1 => ['pipe', 'w'], // stdout
  2 => ['pipe', 'w'], // stderr
];

$proc = proc_open($command, $descriptors, $pipes);
if (!is_resource($proc))
  die("Failed to open process for command $command");

if ($output = stream_get_contents($pipes[1]))
  echo "Output: $output\n\n";
fclose($pipes[1]);

if ($errors = stream_get_contents($pipes[2]))
  echo "Errors: >>>>>>>>\n$errors\n<<<<<<<\n\n";
fclose($pipes[2]);

$exit_code = proc_close($proc);

if ($exit_code != 0) {
  die("Failed to download $url to $output_file");
}

clearstatcache(true, $output_file);
if (!file_exists($output_file)) {
  die("File $output_file not found on file system, command: $command");
}

echo "$url\nhas been saved to $output_file\n";

Выход

Errors: >>>>>>>>
  % Total    % Received % Xferd  Average Speed   Time    Time     Time  Current
                                 Dload  Upload   Total   Spent    Left  Speed
100  2202  100  2202    0     0   4231      0 --:--:-- --:--:-- --:--:--  4735

<<<<<<<

http://php.net/images/logo.php
has been saved to /tmp/default-download-dir/logo.svg

( curl выводит "подробный" вывод в стандартный дескриптор ошибки)

Фиксированная команда

В свете вышесказанного ваша команда должна быть построена следующим образом:

$url = 'http://192.99.8.170:8098/stream/1';
$command = sprintf("curl -o %s %s --max-time $time_in_seconds",
  escapeshellarg($output_file),
  escapeshellarg($url));
Другие вопросы по тегам