FFmpeg (миниатюра из видео) работает в командной строке, но не в PHP-скрипте

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

Позже моя цель — сделать миниатюру из видеофайла, транслируемого в прямом эфире.

Ниже показана моя текущая настройка.

Линукс Убунту 16.04

PHP версии 7.0

Nginx версии 1.10.3

Версия FFmpeg FFmpeg 2.8.11

Я установил FFmpeg с помощью приведенных ниже команд на своем сервере Ubuntu.

apt-get update
apt-get install ffmpeg

Когда я набираю приведенную ниже командную строку в терминале, преобразование выполняется нормально.

ffmpeg -i /etc/nginx/html/test.mp4 -ss 00:00:01 -vframes 1 -vf scale=240:120 /etc/nginx/html/Share/ffmpeg/thumbnails/test.png;

Чтобы использовать его в файле php, я подтвердил расположение двоичных файлов ffmpeg, набрав в терминале which ffmpeg, и он возвращает /usr/bin/ffmpeg

Я также дал разрешение на каталог с помощью командной строки chmod 755.

Вот что у меня есть в моем файле ffmpeg.php. Как видите, я включил полный путь к моему ffmpeg.

<?php
header("Content-Type: text/html;charset=UTF-8");

$ffmpeg = "/usr/bin/ffmpeg";
$videoFile="/etc/nginx/html/test.mp4";
$thumbnail= "/etc/nginx/html/Share/ffmpeg/thumbnails/thumbnail1.jpg";

$cmd = "$ffmpeg -i $videoFile -ss 00:00:01 -vframes 1 -vf scale=240:120 $thumbnail 2>&1 &";

$output = shell_exec($cmd);
echo "<pre>".$output."</pre>";
?>

Это возвращает сообщение, как показано ниже.

ffmpeg version 2.8.11-0ubuntu0.16.04.1 Copyright (c) 2000-2017 the FFmpeg developers
  built with gcc 5.4.0 (Ubuntu 5.4.0-6ubuntu1~16.04.4) 20160609
  configuration: --prefix=/usr --extra-version=0ubuntu0.16.04.1 --build-suffix=-ffmpeg --toolchain=hardened --libdir=/usr/lib/x86_64-linux-gnu --incdir=/usr/include/x86_64-linux-gnu --cc=cc --cxx=g++ --enable-gpl --enable-shared --disable-stripping --disable-decoder=libopenjpeg --disable-decoder=libschroedinger --enable-avresample --enable-avisynth --enable-gnutls --enable-ladspa --enable-libass --enable-libbluray --enable-libbs2b --enable-libcaca --enable-libcdio --enable-libflite --enable-libfontconfig --enable-libfreetype --enable-libfribidi --enable-libgme --enable-libgsm --enable-libmodplug --enable-libmp3lame --enable-libopenjpeg --enable-libopus --enable-libpulse --enable-librtmp --enable-libschroedinger --enable-libshine --enable-libsnappy --enable-libsoxr --enable-libspeex --enable-libssh --enable-libtheora --enable-libtwolame --enable-libvorbis --enable-libvpx --enable-libwavpack --enable-libwebp --enable-libx265 --enable-libxvid --enable-libzvbi --enable-openal --enable-opengl --enable-x11grab --enable-libdc1394 --enable-libiec61883 --enable-libzmq --enable-frei0r --enable-libx264 --enable-libopencv
  libavutil      54. 31.100 / 54. 31.100
  libavcodec     56. 60.100 / 56. 60.100
  libavformat    56. 40.101 / 56. 40.101
  libavdevice    56.  4.100 / 56.  4.100
  libavfilter     5. 40.101 /  5. 40.101
  libavresample   2.  1.  0 /  2.  1.  0
  libswscale      3.  1.101 /  3.  1.101
  libswresample   1.  2.101 /  1.  2.101
  libpostproc    53.  3.100 / 53.  3.100
Input #0, mov,mp4,m4a,3gp,3g2,mj2, from '/etc/nginx/html/Share/test.mp4':
  Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2mp41
    creation_time   : 1970-01-01 00:00:00
    encoder         : Lavf53.24.2
  Duration: 00:00:06.40, start: 0.000000, bitrate: 1321 kb/s
    Stream #0:0(und): Video: mpeg4 (Simple Profile) (mp4v / 0x7634706D), yuv420p, 640x368 [SAR 1:1 DAR 40:23], 932 kb/s, 25 fps, 25 tbr, 12800 tbn, 25 tbc (default)
    Metadata:
      creation_time   : 1970-01-01 00:00:00
      handler_name    : VideoHandler
    Stream #0:1(und): Audio: aac (LC) (mp4a / 0x6134706D), 48000 Hz, 5.1, fltp, 383 kb/s (default)
    Metadata:
      creation_time   : 1970-01-01 00:00:00
      handler_name    : SoundHandler
[swscaler @ 0x1621be0] deprecated pixel format used, make sure you did set range correctly
Output #0, image2, to '/etc/nginx/html/Share/ffmpeg/thumbnails/thumbnail1.jpg':
  Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2mp41
    encoder         : Lavf56.40.101
    Stream #0:0(und): Video: mjpeg, yuvj420p(pc), 240x120 [SAR 20:23 DAR 40:23], q=2-31, 200 kb/s, 25 fps, 25 tbn, 25 tbc (default)
    Metadata:
      creation_time   : 1970-01-01 00:00:00
      handler_name    : VideoHandler
      encoder         : Lavc56.60.100 mjpeg
Stream mapping:
  Stream #0:0 -> #0:0 (mpeg4 (native) -> mjpeg (native))
Press [q] to stop, [?] for help
[image2 @ 0x16165a0] Could not open file : /etc/nginx/html/Share/ffmpeg/thumbnails/thumbnail1.jpg
av_interleaved_write_frame(): Input/output error
frame=    1 fps=0.0 q=3.9 Lsize=N/A time=00:00:00.04 bitrate=N/A    
video:8kB audio:0kB subtitle:0kB other streams:0kB global headers:0kB muxing overhead: unknown
Conversion failed!

ОБНОВЛЕНО: я также дал разрешение на ffmpeg с помощью chmod +x $(который ffmpeg)

Я попробовал php /etc/nginx/html/Share/ffmpeg/ffmpeg.php в терминале, и он также отлично конвертирует.

  libavutil      54. 31.100 / 54. 31.100
  libavcodec     56. 60.100 / 56. 60.100
  libavformat    56. 40.101 / 56. 40.101
  libavdevice    56.  4.100 / 56.  4.100
  libavfilter     5. 40.101 /  5. 40.101
  libavresample   2.  1.  0 /  2.  1.  0
  libswscale      3.  1.101 /  3.  1.101
  libswresample   1.  2.101 /  1.  2.101
  libpostproc    53.  3.100 / 53.  3.100
Input #0, mov,mp4,m4a,3gp,3g2,mj2, from '/etc/nginx/html/Share/test.mp4':
  Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2mp41
    creation_time   : 1970-01-01 00:00:00
    encoder         : Lavf53.24.2
  Duration: 00:00:06.40, start: 0.000000, bitrate: 1321 kb/s
    Stream #0:0(und): Video: mpeg4 (Simple Profile) (mp4v / 0x7634706D), yuv420p, 640x368 [SAR 1:1 DAR 40:23], 932 kb/s, 25 fps, 25 tbr, 12800 tbn, 25 tbc (default)
    Metadata:
      creation_time   : 1970-01-01 00:00:00
      handler_name    : VideoHandler
    Stream #0:1(und): Audio: aac (LC) (mp4a / 0x6134706D), 48000 Hz, 5.1, fltp, 383 kb/s (default)
    Metadata:
      creation_time   : 1970-01-01 00:00:00
      handler_name    : SoundHandler
[swscaler @ 0x72dbe0] deprecated pixel format used, make sure you did set range correctly
Output #0, image2, to '/etc/nginx/html/Share/ffmpeg/thumbnails/thumbnail.jpg':
  Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2mp41
    encoder         : Lavf56.40.101
    Stream #0:0(und): Video: mjpeg, yuvj420p(pc), 240x120 [SAR 20:23 DAR 40:23], q=2-31, 200 kb/s, 25 fps, 25 tbn, 25 tbc (default)
    Metadata:
      creation_time   : 1970-01-01 00:00:00
      handler_name    : VideoHandler
      encoder         : Lavc56.60.100 mjpeg
Stream mapping:
  Stream #0:0 -> #0:0 (mpeg4 (native) -> mjpeg (native))
Press [q] to stop, [?] for help
frame=    1 fps=0.0 q=3.9 Lsize=N/A time=00:00:00.04 bitrate=N/A    
video:8kB audio:0kB subtitle:0kB other streams:0kB global headers:0kB muxing overhead: unknown

person JayB Kim    schedule 02.03.2017    source источник
comment
Вы уверены, что у вас есть разрешение на доступ к /etc/nginx/html/Share/ffmpeg/thumbnails/thumbnail1.jpg ?   -  person Ataur Rahman    schedule 02.03.2017
comment
@Ataur Rahman Да, я дал разрешение на доступ к этому пути. Я установил FFmpeg через apt-get install ffmpeg и все. Использование shell_exec() требует установки ffmpeg-php?   -  person JayB Kim    schedule 02.03.2017
comment
Я не думаю, что это необходимо. На моем сервере я также использую ffmpeg от shell_exec, никаких проблем. Хотя вы упомянули о предоставлении разрешения на папку, я все же думаю, что есть какая-то проблема с разрешением. Можете ли вы попытаться создать тестовый файл по этому пути, используя file_get_contents? Давайте посмотрим, возвращает ли он какую-либо ошибку.   -  person Ataur Rahman    schedule 02.03.2017
comment
@Ataur Rahman Я только что создал test.txt (Привет!) В каталоге ~/thumbnails. И в моем файле test.php я написал $dir = file_get_contents('/etc/nginx/html/Share/ffmpeg/thumbnails/test.txt'); echo "test->".$dir; Это не возвращает ошибку, test->Hello There!, так что я думаю, что разрешение в порядке. Это действительно странно...   -  person JayB Kim    schedule 02.03.2017
comment
Я также проверил, правильно ли работает shell_exec(). $cmd = shell_exec('ls'); echo "cmd->".$cmd; и он возвращается нормально.   -  person JayB Kim    schedule 02.03.2017
comment
о, мой плохой, вы можете попробовать file_put_contents создать файл, я ранее упомянул неправильную функцию!   -  person Ataur Rahman    schedule 02.03.2017
comment
@AtaurRahman О, ладно, file_put_contents не работает. Несмотря на то, что я указал путь chmod -R 777, он недоступен для записи. странно... Warning: file_put_contents(/etc/nginx/html/Share/ffmpeg/thumbnails/test.txt): failed to open stream:   -  person JayB Kim    schedule 02.03.2017
comment
@AtaurRahman Я выполнил следующую команду chown www-data:www-data '/etc/nginx/html/' (более короткий путь), и touch /etc/nginx/html/thumbnail.jpg работает нормально. Я был счастлив увидеть результат. Однако этот путь /etc/nginx/html/Share/ffmpeg/thumbnails/ не работает. Как вы думаете, это может вызвать проблемы с разрешением? Share folder — это символическая ссылка, которую я сделал из /media/Share/, которая использует общую папку с моей основной ОС.   -  person JayB Kim    schedule 02.03.2017


Ответы (2)


Похоже, у вас нет прав доступа к каталогу
/etc/nginx/html/Share/ffmpeg/thumbnails/

По крайней мере, это то, что я получаю из сообщения об ошибке Could not open file : /etc/nginx/html/Share/ffmpeg/thumbnails/thumbnail1.jpg.

Быстрый и грязный способ проверить, является ли это причиной проблемы, состоит в том, чтобы изменить уровень разрешений каталогов на 777. (Конечно, верните это позже)

person Tek    schedule 02.03.2017
comment
Я только что дал chmod 777 /etc/nginx/html/Share/ffmpeg/thumbnails/, и он все равно не работает. - person JayB Kim; 02.03.2017
comment
@JunKim Вы получаете такое же сообщение об ошибке? Возможно ли, что файл thumbnail1.jpg уже существует из предыдущего теста с вашей учетной записью пользователя, и у веб-пользователя нет прав на изменение файла? (вы можете попробовать выполнить chmod -R 777 в каталоге) - person Tek; 02.03.2017
comment
Да, все еще с тем же сообщением об ошибке. Я удалил все проверенные файлы изображений и попробовал снова. Все то же самое... Что это может быть за проблема...? Насколько я могу судить, shell_exec() выполняется правильно, поскольку я тестировал базовую команду Linux, ffmpeg установлен правильно, и разрешение на путь уже предоставлено с помощью chmod 777. Что ж, надеюсь, я смогу решить эту проблему. в течение нескольких часов. - person JayB Kim; 02.03.2017
comment
@JunKim Следующие шаги, которые я бы предпринял, были бы следующими: 1. Измените скрипт php, чтобы просто создать пустой файл в /etc/nginx/html/Share/ffmpeg/thumbnails/thumbnail1.jpg, то есть что-то вроде: $cmd = "touch $thumbnail"; и проверить, успешно ли создан файл (это должно исключить любые проблемы с разрешениями) . 2. Выполните php-скрипт от имени пользователя веб-сервера из командной строки и проверьте результаты. - person Tek; 02.03.2017
comment
Я думал об этом какое-то время. Есть ли что-нибудь, что я должен был сделать, чтобы связать php7.0 с ffmpeg? - person JayB Kim; 02.03.2017
comment
@JunKim Кто-нибудь поправит меня, если я ошибаюсь, но я так не думаю, иначе сценарий полностью выйдет из строя (ffmpeg не может быть выполнен). Также это не имеет ничего общего с PHP, потому что с помощью функции shell_execute вы делегируете системе исполняемую команду. - person Tek; 02.03.2017
comment
Я просто поместил этот код $thumbnail = "/etc/nginx/html/Share/ffmpeg/thumbnails/thumbnail1.jpg"; $cmd = "touch $thumbnail"; echo shell_exec($cmd); в файл php. Когда я выполнил его, он не создает никаких файлов в этом каталоге... (из терминала файл создается успешно) Так что, я думаю, это должно быть проблемой с разрешением? - person JayB Kim; 02.03.2017
comment
@JunKim Да, точно, так что это должно быть проблемой разрешения. Пожалуйста, выполните следующую команду, а затем повторите попытку сценария: chmod -R 777 /etc/nginx/html/Share/ffmpeg/thumbnails/ . Мне очень любопытен результат! - person Tek; 02.03.2017
comment
Я немного отдохнул и вернулся. Я дал разрешение chmod -R 777 на этот путь, и он все еще не работает. Мне действительно любопытно, почему... это недоступно для записи. Я тестировал более короткий путь touch /etc/nginx/html/thumbnail.jpg, и он тоже работает. - person JayB Kim; 02.03.2017
comment
Я выполнил следующую команду chown www-data:www-data '/etc/nginx/html/' (более короткий путь), и touch /etc/nginx/html/thumbnail.jpg работает нормально. Я был счастлив увидеть результат. Однако этот путь /etc/nginx/html/Share/ffmpeg/thumbnails/ не работает. Как вы думаете, это может вызвать проблемы с разрешением? Share folder — это символическая ссылка, которую я сделал из /media/Share/, которая использует общую папку с моей основной ОС. - person JayB Kim; 02.03.2017
comment
Без параметра -R команда chown изменяет разрешения только для папки, в которой она выполняется, но не для подкаталогов. Это означает, что ваш /etc/nginxs/html/ теперь должен принадлежать www-data, но похоже, что это не так для /etc/nginx/html/Share/ffmpeg/thumbnails/. Что касается символической ссылки: я не уверен, но вполне возможно, что это немного усложняет дело. Что бы я сделал, так это выполнить chown -R www-data:www-data '/etc/nginx/html/', чтобы это применялось ко всем подпапкам, и проверить, решает ли это уже проблему. - person Tek; 02.03.2017
comment
Спасибо большое за помощь. StackOverFlow — отличный сайт! Я организовал решение ниже, касающееся проблемы с разрешениями и символической ссылки. Мне пришлось дать доступ на запись/чтение к общей папке в операционной системе хоста. - person JayB Kim; 02.03.2017

Я РЕШИЛ вышеуказанную проблему с большой помощью от @Tek и @AtaurRahman.

Это была проблема с разрешением

Пришлось выполнять команды chown/chmod по пути

chmod -R 777 'path';
chown -R www-data:www-data 'path';

команда chown: изменяет права пользователя и/или группы на данный файл.

команда chmod: изменяет права доступа к файлу (доступ на чтение и запись).

У меня также была проблема с разрешением на совместное использование папки с моей операционной системой хоста.

Как вы можете видеть мой вопрос выше, /etc/nginx/html/Share/~ находится там, где

Я хотел написать файл .jpg. Я дал доступ чтение/запись всем к общей папке в моей основной ОС, все работает отлично. (Я не уверен, что это вызовет какие-либо проблемы с безопасностью позже. Если кто-то знает об этом, не стесняйтесь комментировать ниже, чтобы поделиться)

Я надеюсь, что это поможет тем, кто сталкивается с той же проблемой

person JayB Kim    schedule 02.03.2017