无法使用Node js在AWS Lambda函数中将视频文件转换为音频文件

时间:2019-02-20 19:49:23

标签: node.js amazon-web-services ffmpeg aws-lambda

我无法使用Node JS在AWS lambda函数内部将视频文件转换为音频文件。在运行我的lambda函数时,它不会抛出任何错误,它执行时不会出现任何错误。但是音频文件大小仍为0 MB。我无法在代码中找到错误或任何问题。

这是我的代码,

const fs = require('fs');
const childProcess = require('child_process');
const AWS = require('aws-sdk');
const path = require('path');
AWS.config.update({
    region : 'us-east-2'
});
const s3 = new AWS.S3({apiVersion: '2006-03-01'});


exports.handler = (event, context, callback) => {
    process.env.PATH = process.env.PATH + ':/tmp/';
    process.env['FFMPEG_PATH'] = '/tmp/ffmpeg';
    const BIN_PATH = process.env['LAMBDA_TASK_ROOT'];
    process.env['PATH'] = process.env['PATH'] + ':' + BIN_PATH;

    childProcess.exec(
        'cp /var/task/ffmpeg /tmp/.; chmod 755 /tmp/ffmpeg;',
        function (error, stdout, stderr) {
            if (error) {
                console.log('Error occured',error);
            } else {
                var ffmpeg = '/tmp/ffmpeg';
                var createStream = fs.createWriteStream("/tmp/video.mp3");
                createStream.end();
                var params = {
                    Bucket: "test-bucket",
                    Key: event.Records[0].s3.object.key
                };
                s3.getObject(params, function(err, data) {
                    if (err) {
                        console.log("Error", err);
                    }
                    fs.writeFile("/tmp/vid.mp4", data.Body, function (err) {
                        if (err) console.log(err.code, "-", err.message);
                        return callback(err);
                    }, function() {
                        try {
                            var stats = fs.statSync("/tmp/vid.mp4");
                            console.log("size of the file1 ", stats["size"]);
                            try {
                                console.log("Yeah");
                                const inputFilename = "/tmp/vid.mp4";
                                const mp3Filename = "/tmp/video.mp3";
                                // // Convert the FLV file to an MP3 file using ffmpeg.
                                const ffmpegArgs = [
                                    '-i', inputFilename,
                                    '-vn', // Disable the video stream in the output.
                                    '-acodec', 'libmp3lame', // Use Lame for the mp3 encoding.
                                    '-ac', '2', // Set 2 audio channels.
                                    '-q:a', '6', // Set the quality to be roughly 128 kb/s.
                                    mp3Filename,
                                ];
                                try {
                                    const process = childProcess.spawnSync(ffmpeg, ffmpegArgs);
                                    console.log("stdout ", process.stdout);
                                    console.log("stderr ", process.stderr);
                                    console.log("tmp files ");
                                    fs.readdir('/tmp/', (err, files) => {
                                        files.forEach(file => {
                                            var stats = fs.statSync(`/tmp/${file}`);
                                            console.log("size of the file2 ", stats["size"]);
                                          console.log(file);
                                        });
                                      });

                                } catch (e) {
                                    console.log("error while converting video to audio ", e);
                                }

                                // return process;
                            } catch (e) {
                                console.log(e);
                            }
                        } catch (e) {
                            console.log("file is not complete", e);
                        }
                    }, function () {
                        console.log("checking ");
                        var stats = fs.statSync("/tmp/video.mp3");
                        console.log("size of the file2 ", stats["size"]);
                    });

                    return callback(err);
                });
            }
        }
    )
}

代码工作流程

首先,我下载了ffmpeg二进制exec文件并放入了我的项目目录。之后,我压缩了项目并将其放入lambda函数。每当将新文件上传到S3存储桶时,都会触发该lambda函数。我检查了/ tmp /存储文件和存在的音频文件.mp3,但大小为0 MB。

注意

而且,在我的代码中,以下内容未调用或此部分未达到。当我查看Cloudwatch日志时,看不到此控制台日志消息。我不知道为什么这个函数没有调用。

function () {
        console.log("checking ");
        var stats = fs.statSync("/tmp/video.mp3");
        console.log("size of the file2 ", stats["size"]);
    });

请帮助我找到此问题的解决方案。我花了很多时间弄清楚这个问题。但是我找不到解决方案。欢迎任何建议! 谢谢,

1 个答案:

答案 0 :(得分:0)

有很多局限性可能导致Lambda在尝试进行转换时炸毁。您需要做的第一件事是为AWS Linux编译ffmpeg,通常,您必须使用静态链接而不是动态链接进行编译。

另一种方法是使用docker容器并在aws ecs fargate上运行它,这将使您更轻松地控制依赖项,并且对运行时间没有任何限制,并且您仍然可以外包机器管理到AWS。

在Lambda上对视频进行转码 https://intoli.com/blog/transcoding-on-aws-lambda/

预编译的ffmpeg https://johnvansickle.com/ffmpeg/