Electron录制应用-打包静态文件问题【命令行ffmpeg导不出视频】

问题描述

在开发环境下,所有功能都运行正常,但一旦进行打包并运行生产环境的版本,导出mp4视频的功能就失效了。没有文件生成,也没有任何错误提示。

排查问题

为了找到问题的根源,我首先决定通过日志来追踪。我使用了winston和winston-daily-rotate-file这两个强大的日志库来记录应用程序的运行信息。通过分析日志,我发现了一个关键的线索:

ffmpeg(一个用于处理多媒体内容的开源库)没有被正确地打包进最终的产品中。

复制代码
const { createLogger, format, transports } = require("winston");
require("winston-daily-rotate-file");
const { app } = require('electron');
const path = require('path');
const fs = require('fs');

const logDir = path.join(app.getPath('userData'), 'logs');

if (!fs.existsSync(logDir)) {
  try {
    fs.mkdirSync(logDir);
  } catch (error) {
    console.error('Failed to create logs directory:', error);
  }
}


const customFormat = format.combine(
    format.timestamp({ format: "MMM-DD-YYYY HH:mm:ss" }),
    format.align(),
    format.printf((i: { level: any; timestamp: any; message: any; }) => `${i.level}: ${[i.timestamp]}: ${i.message}`)
);
const defaultOptions = {
    format: customFormat,
    datePattern: "YYYY-MM-DD",
    zippedArchive: true,
    maxSize: "20m",
    maxFiles: "14d",
};
const globalLogger = createLogger({
    format: customFormat,
    transports: [
        new transports.DailyRotateFile({
            filename: logDir + "/info-%DATE%.log",
            level: "info",
            ...defaultOptions,
        }),
        new transports.DailyRotateFile({
            filename:  logDir + "/error-%DATE%.log",
            level: "error",
            ...defaultOptions,
        }),
    ],
});

export default globalLogger

尝试解决(方案一:失败)

我首先尝试使用webpack的copy-webpack-plugin插件将ffmpeg复制到dist/main/bin目录下。然而,即使这样做了,问题依然没有得到解决。

成功解决(方案二)

经过深思熟虑,我意识到问题的关键在于如何让打包后的程序能够找到ffmpeg文件。于是,我采取了以下步骤:

放置ffmpeg文件:我将ffmpeg的文件放置在项目的assets文件夹中。

配置打包参数:在package.json文件中,我配置了extraResources选项,以确保assets文件夹中的文件能够被打包进去。

复制代码
// package.json 配置extraResources  
"extraResources": [  
  "./assets/**"  
]

获取ffmpeg的路径:在代码中,我根据运行平台(如Windows、Mac或Linux)和架构(如x64或arm64)动态地生成ffmpeg的路径。

复制代码
// 查找assets中ffmpeg的输出路径  
var os = require('os');  
var path = require('path');  
  
var platform = os.platform();  
// 适配electron-builder的智能构建过程  
if (platform === 'darwin') {  
  platform = 'mac';  
} else if (platform === 'win32') {  
  platform = 'win';  
}  
  
// ...(其他平台判断和架构检查代码省略)  
  
var ffmpegPath = (dirname: string) => path.join(  
  dirname,  
  'ffmpeg-exe',  
  'bin',  
  platform,  
  arch,  
  platform === 'win' ? 'ffmpeg.exe' : 'ffmpeg'  
);  
export default ffmpegPath;

执行命令行:最后,我使用Node.js的child_process模块中的exec函数来执行ffmpeg的命令行。

复制代码
const { exec } = require('child_process');  
exec(command);

注意事项

在生成日志文件夹时,我遇到了读写权限的问题。为了避免这个问题,我使用了Electron的app.getPath('userData')方法来获取应用程序内部的文件路径,这些路径通常都具有读写权限。

复制代码
const { app } = require('electron');  
const logDir = path.join(app.getPath('userData'), 'logs');

总结

一直要有日志!!

相关推荐
檀越剑指大厂17 分钟前
【Nginx系列】Tengine:基于 Nginx 的高性能 Web 服务器与反向代理服务器
服务器·前端·nginx
是你的小橘呀30 分钟前
深入理解 JavaScript 预编译:从原理到实践
前端·javascript
uhakadotcom33 分钟前
在使用cloudflare workers时,假如有几十个请求,如何去控制并发?
前端·面试·架构
风止何安啊35 分钟前
栈与堆的精妙舞剧:JavaScript 数据类型深度解析
前端·javascript
用户47949283569151 小时前
Chrome DevTools MCP:让 AI 助手直接操作浏览器开发工具
前端·javascript·chrome
Rysxt_1 小时前
Vuex 教程 从入门到实践
前端·javascript·vue.js
开开心心就好1 小时前
微软官方出品:免费数据恢复工具推荐
网络·笔记·microsoft·pdf·word·音视频·symfony
by__csdn1 小时前
Node.js版本与npm版本的对应关系
前端·npm·node.js
AI_56782 小时前
Webpack性能优化终极指南:4步实现闪电打包
前端·webpack·性能优化
懷淰メ2 小时前
python3GUI--短视频社交软件 By:Django+PyQt5(前后端分离项目)
后端·python·django·音视频·pyqt·抖音·前后端