从Fluent-Ffmpeg Api调用Ffmpeg二进制文件时节点中的ENOENT错误

时间:2017-08-14 21:01:44

标签: node.js firebase ffmpeg firebase-storage fluent-ffmpeg

背景

我正在连接节点中的firebase功能。目的是将入站音频剪辑解析为设定长度。使用ffmpeg和fluent-ffmpeg。

问题

当在firebase中触发该函数时,当Fluent-Ffmpeg尝试访问Ffmpeg二进制文件时,我收到ENOENT错误

Firebase调试输出

  

错误:{错误:产生   ./Cloud/functions/node_modules/ffmpeg-binaries/bin/ffmpeg ENOENT       at exports._errnoException(util.js:1018:11)       在Process.ChildProcess._handle.onexit(internal / child_process.js:193:32)       at onErrorNT(internal / child_process.js:367:16)       at _combinedTickCallback(internal / process / next_tick.js:80:11)       at process._tickDomainCallback(internal / process / next_tick.js:128:9)代码:' ENOENT',errno:   ' ENOENT',系统调用:' spawn   ./Cloud/functions/node_modules/ffmpeg-binaries/bin/ffmpeg' ;,路径:   ' ./云/功能/ node_modules / ffmpeg的-二进制/ bin中/ ffmpeg的&#39 ;,
  spawnargs:[' -formats' ]}

预期结果

入站文件被下载到临时目录,被裁剪,并作为裁剪文件重新上传到firebase存储。

环境

  • mac客户端/ firebase存储
  • node v8.1.0
  • ffmpeg v3.2.2
  • fluent-ffmpeg v2.1.2

代码[更新以反映Svenskunganka的变化。现在工作]

const ffmpeg = require('fluent-ffmpeg');
const PREVIEW_PREFIX = 'preview_';

exports.generatePreviewClip = functions.storage.object('audioFiles').onChange(event => {

      //console.log('Times this function has run: ', run++);

      const object = event.data; // The Storage object.
      const fileBucket = object.bucket; // The Storage bucket that contains the file.
      const filePath = object.name; // File path in the bucket.
      const contentType = object.contentType; // File content type.
      const resourceState = object.resourceState; // The resourceState is 'exists' or 'not_exists' (for file/folder deletions).
      const metageneration = object.metageneration; // Number of times metadata has been generated. New objects have a value of 1.

      // Exit if this is triggered on a file that is not an audio file.
      if (!contentType.startsWith('audio/')) {
        console.log('This is not an audio file.');
        console.log('This is the file:', filePath);
        return;
      }

      // Get the file name.
      const fileName = path.basename(filePath);
      console.log('Working with filename', fileName);
      // Exit if the file is already an audio clip.
      if (fileName.startsWith(PREVIEW_PREFIX)) {
        console.log('Already a preview clip.');
        return;
      }

      // Exit if this is a move or deletion event.
      if (event.data.resourceState === 'not_exists') {
        console.log('This is a deletion event.');
        return;
      }

      // Exit if file exists but is not new and is only being triggered
      // because of a metadata change.
      if (resourceState === 'exists' && metageneration > 1) {
        console.log('This is a metadata change event.');
        return;
      }

      // Download file from bucket.

      const bucket = gcs.bucket(fileBucket);
      const tempFilePath = path.join(os.tmpdir(), fileName);
      return bucket.file(filePath).download({
        destination: tempFilePath
      }).then(() => {

        console.log('Audio file downloaded locally to temp directory', tempFilePath);

    var ffmpegPath = require("ffmpeg-binaries").ffmpegPath();
    var ffprobePath = require("ffmpeg-binaries").ffprobePath();

    // Generate a croped file using ffmpeg.
    var command = new ffmpeg(tempFilePath);
        command.setFfmpegPath(ffmpegPath);
        command.setFfprobePath(ffprobePath);

        command
              .setStartTime('00:00:03')
              .setDuration('10')
              .output(tempFilePath)
              .on('end', function() {
                    console.log('Audio Crop Done Successfully');
               })
               .on('error', function(err)
               {
                  console.log('Error:', err);
               }).run();

              }).then(() => {
        console.log('Preview file created at', tempFilePath);
        // We add a 'preview_' prefix to the audio file name. that's how it will appear in firebase.
        const previewFileName = PREVIEW_PREFIX + fileName;
        console.log('previewFileName is', previewFileName)
        const previewFilePath = path.join(path.dirname(filePath), previewFileName);
        console.log('previewFilePath is', previewFilePath);
        // Uploading the preview file.
        return bucket.upload(tempFilePath, {destination: previewFilePath});
      // Once the file has been uploaded delete the local file to free up disk space.
      }).then(() => fs.unlinkSync(tempFilePath));

      // [END audio file generation]

    });

我的ffmpeg-binaries / bin目录的内容和结构

-rwxrwxrwx  1 sherpa  staff    24M Dec 10  2016 ffmpeg
-rwxr--r--  1 sherpa  staff    35M Jan 12  2017 ffmpeg.exe
-rwxr--r--  1 sherpa  staff    35M Jan 12  2017 ffplay.exe
-rwxrwxrwx  1 sherpa  staff    24M Dec 10  2016 ffprobe
-rwxr--r--  1 sherpa  staff    35M Jan 12  2017 ffprobe.exe
-rwxrwxrwx  1 sherpa  staff    22M Dec 10  2016 ffserver

我尝试过的事情

  • 我可以从命令行执行ffmpeg
  • sudo chmod -R u + x ffmpeg-binaries /
  • ffmpeg在全局路径中设置
  • 在setFfmpegPath中使用了ffmpeg.exe二进制文件,结果相同
    • 错误:{错误:产生./Cloud/functions/node_modules/ffmpeg-binaries/bin/ffmpeg.exe ENOENT
  • 使用了许多不同的setFfmpegPath路径结构,例如:
    • ./云/功能/ node_modules / ffmpeg的-二进制/ bin中/ ffmpeg的
    • node_modules / ffmpeg的-二进制/ bin中/ ffmpeg的
    • ./云/功能/ node_modules / ffmpeg的-二进制/ bin中/

感谢您的任何建议。

1 个答案:

答案 0 :(得分:0)

我们在针对该问题的评论中解决了该问题,但我会针对可能存在相同问题的未来用户发布答案。问题是提供给setFfmpegPath()方法的路径是相对的,而应该是绝对的。 ffmpeg-binaries模块导出一对辅助函数,您可以调用它们来获取其二进制文件的路径:

var ffmpeg = require("fluent-ffmpeg")
var ffmpegPath = require("ffmpeg-binaries")

ffmpeg
  .setFfmpegPath(ffmpegPath)
  ...

确保ffmpeg-binaries安装了npm i -S ffmpeg-binaries

2018年11月7日更新:

ffmpeg-binaries软件包在版本4.0.0中发布了一个新的重大更改,删除了它导出的所有函数,而只是导出了一个指向ffmpeg目录的字符串。这在提交009e4d5中已更改 我已更新答案以反映这些变化。