Почему PHP зависает после записи 4096 байт в процесс, начатый с proc open?
для тех, кому интересно, после того, как он оставил все это на пару часов, он теперь работает отлично.
Я пытаюсь передать видеофайл VLC, используя PHP в качестве доказательства концепции для предстоящего предложения проекта для кого-то.
мне удалось показать, что он работает, создав файл
вот пример сценария, чтобы увидеть, что я имею в виду:
$filepath = 'Path/to/your/video';
$vlcpath = 'Path/to/your/VLC executable';
$descriptorspec = array(
0 => array("pipe", "r"), // stdin
1 => array("pipe", "w"), // stdout
2 => array("pipe", "w") // stderr
);
$vlc = proc_open($vlcpath . ' -', $descriptorspec, $pipes, null, null, ['bypass_shell' => true]);
$file = fopen($filepath, 'r');
stream_copy_to_stream($file, $pipes[0]);
fclose($file);
proc_close($vlc);
Я на Windows 10 и использую PHP 5.5.31. Я видел несколько отчетов об ошибках на сайте PHP об этом, но они предполагают, что последняя версия исправила его. Я не совсем понимаю концепции блокировки потока, но я уже пробовал PHP v7.0.3 безрезультатно.
Я запускаю этот скрипт с помощью командной строки:php file.php
1 ответов
я столкнулся с той же проблемой, пытаясь сделать преобразование WAV в MP3, используя LAME в Windows, и не смог найти работоспособное решение.
Я пробовал десятки вещей, включая блокировку / неблокирующую запись, запись небольших (
в конечном итоге я сдался, и, к счастью, LAME мог читать входные данные из файла вместо STDIN, поэтому я просто решил записать данные во временный файл, вызвать LAME и удалить временный файл.
вот соответствующий код:
// file descriptors for reading and writing to the Lame process
$descriptors = array(
0 => array('pipe', 'r'), // stdin
1 => array('pipe', 'w'), // stdout
2 => array('pipe', 'a'), // stderr
);
if (strtoupper(substr(PHP_OS, 0, 3)) === 'WIN') {
// workaround for Windows conversion
// writing to STDIN seems to hang indefinitely after writing approximately 0xC400 bytes
$wavinput = tempnam(sys_get_temp_dir(), 'wav');
if (!$wavinput) {
throw new Exception('Failed to create temporary file for WAV to MP3 conversion');
}
file_put_contents($wavinput, $data);
$size = 0;
} else {
$wavinput = '-'; // stdin
}
// Mono, variable bit rate, 32 kHz sampling rate, read WAV from stdin, write MP3 to stdout
$cmd = sprintf("%s -m m -v -b 32 %s -", self::$lame_binary_path, $wavinput);
$proc = proc_open($cmd, $descriptors, $pipes);
if (!is_resource($proc)) {
throw new Exception('Failed to open process for MP3 encoding');
}
stream_set_blocking($pipes[0], 0); // set stdin to be non-blocking
for ($written = 0; $written < $size; $written += $len) {
// write to stdin until all WAV data is written
$len = fwrite($pipes[0], substr($data, $written, 0x20000));
if ($len === 0) {
// fwrite wrote no data, make sure process is still alive, otherwise wait for it to process
$status = proc_get_status($proc);
if ($status['running'] === false) break;
usleep(25000);
} else if ($written < $size) {
// couldn't write all data, small pause and try again
usleep(10000);
} else if ($len === false) {
// fwrite failed, should not happen
break;
}
}
fclose($pipes[0]);
$data = stream_get_contents($pipes[1]);
$err = trim(stream_get_contents($pipes[2]));
fclose($pipes[1]);
fclose($pipes[2]);
$return = proc_close($proc);
if ($wavinput != '-') unlink($wavinput); // delete temp file on Windows
if ($return !== 0) {
throw new Exception("Failed to convert WAV to MP3. Shell returned ({$return}): {$err}");
} else if ($written < $size) {
throw new Exception('Failed to convert WAV to MP3. Failed to write all data to encoder');
}
return $data;