This library provides FFmpeg builds ported to JavaScript using Emscripten project. Builds are optimized for in-browser use: minimal size for faster loading, asm.js, performance tunings, etc. Though they work in Node as well.
Currently available builds (additional builds may be added in future):
ffmpeg-webm.js
- WebM encoding (VP8 & Opus encoders, popular decoders).ffmpeg-worker-webm.js
- Web Worker version offfmpeg-webm.js
.ffmpeg-mp4.js
- MP4 encoding (H.264 & AAC & MP3 encoders, popular decoders).ffmpeg-worker-mp4.js
- Web Worker version offfmpeg-mp4.js
.ffmpeg-worker-hls.js
- HLS (HTTP Live Streaming) in a Web Worker. Data is MPEG2-TS encoded and POSTed via HTTP. Note this uses Web Assembly.ffmpeg-worker-dash.js
- DASH (Dynamic Adaptive Streaming over HTTP) in a Web Worker. Data is POSTed via HTTP. Note this uses Web Assembly.ffmpeg-worker-mkve.js
- Build for mkv-extract without any encoders nor decoders, just muxing. Note this uses Web Assembly.
Note: only NPM releases contain abovementioned files.
ffmpeg.js uses the following version pattern: major.minor.9ddd
, where:
- major - FFmpeg's major version number used in the builds.
- minor - FFmpeg's minor version.
- ddd - ffmpeg.js own patch version. Should not be confused with FFmpeg's patch version number.
Example: 2.7.9005
See documentation on Module object for the list of options that you can pass.
ffmpeg.js provides common module API, ffmpeg-webm.js
is the default module. Add its name after the slash if you need another build, e.g. require("ffmpeg.js/ffmpeg-mp4.js")
.
const ffmpeg = require("ffmpeg.js");
let stdout = "";
let stderr = "";
// Print FFmpeg's version.
ffmpeg({
arguments: ["-version"],
print: function(data) { stdout += data + "\n"; },
printErr: function(data) { stderr += data + "\n"; },
onExit: function(code) {
console.log("Process exited with code " + code);
console.log(stdout);
console.log(stderr);
},
});
Use e.g. browserify in case of Browser.
ffmpeg.js also provides wrapper for main function with Web Worker interface to offload the work to a different process. Worker sends the following messages:
{type: "ready"}
- Worker loaded and ready to accept commands.{type: "run"}
- Worker started the job.{type: "stdout", data: "<line>"}
- FFmpeg printed to stdout.{type: "stderr", data: "<line>"}
- FFmpeg printed to stderr.{type: "exit", data: "<code>"}
- FFmpeg exited.{type: "done", data: "<result>"}
- Job finished with some result.{type: "error", data: "<error description>"}
- Error occurred.{type: "abort", data: "<abort reason>"}
- FFmpeg terminated abnormally (e.g. out of memory, wasm error).- For HLS and DASH:
{type: "start-stream"} - FFmpeg is ready to stream data (application should start posting
stream-data` messages).- `{type: "sending"} - Sent once when data starts to be POSTed.
{type: "ffexit", code: "<code>"}
- FFmpeg exited with status code.{type: "upload", stream: "<chunk as ReadableStream>", url: "<chunk filename>"}
- Only if upload URL starts withpostMessage:
(see below).
You can send the following messages to the worker:
{type: "run", ...opts}
- Start new job with provided options.- For HLS, you should typically send something based on this:
type: 'run', arguments: [ '-seekable', '0', '-loglevel', 'info', '-i', '/work/stream1', '-map', '0:v', '-map', '0:a', '-c:v', 'copy', // assumes video is already in desired encoding '-c:a', 'copy', // assumes audio is already in desired encoding '-f', 'hls', // use hls encoder '-hls_time', '2', // 2 second HLS chunks '-hls_segment_type', 'mpegts', // MPEG2-TS muxer '-hls_list_size', '2', // two chunks in the list at a time '-hls_flags', 'split_by_time', // if you don't have < 2s keyframes '/outbound/output.m3u8' // path to media playlist file in virtual FS, // must be under /outbound ], MEMFS: [ { name: 'stream1' }, { name: 'stream2' } ]
- For HLS, you should typically send something based on this:
* For DASH, you should typically send something based on this:
```js
type: 'run',
arguments: [
'-seekable', '0',
'-loglevel', 'info',
'-i', '/work/stream1',
'-map', '0:v',
'-map', '0:a',
'-c:v', 'copy', // assumes video is already in desired encoding
'-c:a', 'copy', // assumes audio is already in desired encoding
'-f', 'dash', // use dash encoder
'-seg_duration', '2', // 2 second segments
'-window_size', '2', // two chunks in the list at a time
'-streaming', '1', // fragment data
'-dash_segment_type', 'webm', // container type
'/outbound/output.mpd' // path to manifest file in virtual FS,
// must be under /outbound
],
MEMFS: [
{ name: 'stream1' },
{ name: 'stream2' }
]
- For HLS and DASH:
{type: "base-url", data: "<upload-url>", protocol: "hls|dash", options: "<fetch request options>}
- Sets the URL to stream data to. The generated HLS or DASH chunk filenames are appended to this. Send this before sending anystream-data
. Default method is POST but you can change this using the request options.- Note: If the upload URL starts with
postMessage:
then instead of streaming to the network, the Worker willpostMessage
each chunk (see above).
- Note: If the upload URL starts with
{type: "stream-data", name: "<filename>", data: "<data>"}
- Data (audio/video/muxed) to supply to FFmpeg.filename
must match an argument passed to FFmpeg via a-i
option (up to two files are supported).data
is anArrayBuffer
.{type: "stream-end"}
- End all input streams (FFmpeg will exit after this).
const worker = new Worker("ffmpeg-worker-webm.js");
worker.onmessage = function(e) {
const msg = e.data;
switch (msg.type) {
case "ready":
worker.postMessage({type: "run", arguments: ["-version"]});
break;
case "stdout":
console.log(msg.data);
break;
case "stderr":
console.log(msg.data);
break;
case "done":
console.log(msg.data);
break;
}
};
You can use worker_threads module in case of Node.
Empscripten supports several types of file systems. ffmpeg.js uses MEMFS to store the input/output files in FFmpeg's working directory. You need to pass Array of Object to MEMFS
option with the following keys:
- name (String) - File name, can't contain slashes.
- data (ArrayBuffer/ArrayBufferView/Array) - File data.
ffmpeg.js resulting object has MEMFS
option with the same structure and contains files which weren't passed to the input, i.e. new files created by FFmpeg.
const ffmpeg = require("ffmpeg.js");
const fs = require("fs");
const testData = new Uint8Array(fs.readFileSync("test.webm"));
// Encode test video to VP8.
const result = ffmpeg({
MEMFS: [{name: "test.webm", data: testData}],
arguments: ["-i", "test.webm", "-c:v", "libvpx", "-an", "out.webm"],
});
// Write out.webm to disk.
const out = result.MEMFS[0];
fs.writeFileSync(out.name, Buffer(out.data));
You can also mount other FS by passing Array of Object to mounts
option with the following keys:
- type (String) - Name of the file system.
- opts (Object) - Underlying file system options.
- mountpoint (String) - Mount path, must start with a slash, must not contain other slashes and also the following paths are blacklisted:
/tmp
,/home
,/dev
,/work
. Mount directory will be created automatically before mount.
See documentation of FS.mount for more details.
const ffmpeg = require("ffmpeg.js");
ffmpeg({
// Mount /data inside application to the current directory.
mounts: [{type: "NODEFS", opts: {root: "."}, mountpoint: "/data"}],
arguments: ["-i", "/data/test.webm", "-c:v", "libvpx", "-an", "-y", "/data/out.webm"],
});
// out.webm was written to the current directory.
It's recommended to use Docker to build ffmpeg.js.
-
Clone ffmpeg.js repository with submodules:
git clone https://github.com/qgustavor/ffmpeg.js.git --recurse-submodules
-
Modify Makefile and/or patches if you wish to make a custom build.
-
Build everything:
docker run --rm -it -v /path/to/ffmpeg.js:/mnt -w /opt kagamihi/ffmpeg.js # cp -a /mnt/{.git,build,Makefile} . && source /root/emsdk/emsdk_env.sh && make && cp ffmpeg*.{js,wasm} /mnt
That's it. ffmpeg.js modules should appear in your repository clone.
Ubuntu example:
sudo apt-get update
sudo apt-get install -y git python3 build-essential automake libtool pkg-config
cd ~
git clone https://github.com/emscripten-core/emsdk.git && cd emsdk
./emsdk install 3.1.49
./emsdk activate 3.1.49
source emsdk_env.sh
cd ~
git clone https://github.com/qgustavor/ffmpeg.js.git --recurse-submodules && cd ffmpeg.js
make
Thanks to videoconverter.js for inspiration. And of course to all great projects which made this library possible: FFmpeg, Emscripten, asm.js, node.js and many others.
Own library code licensed under LGPL 2.1 or later.
These builds use the LGPL version of FFmpeg and are thus available under LGPL 2.1 or later. See here for more details and FFmpeg's license information.
Included libraries:
- libopus licensed under BSD.
- libvpx licensed under BSD.
See LICENSE.WEBM for the full text of software licenses used in this build.
This build uses GPL version of FFmpeg and is thus available under GPL 2.0. It also includes patent encumbered H.264, AAC and MP3 encoders. Make sure to contact a lawyer before using it in your country.
Included libraries:
- x264 licensed under GPL.
- LAME licensed under LGPL.
See LICENSE.MP4 for the full text of software licenses used in this build.
This build uses the LGPL version of FFmpeg and are thus available under LGPL 2.1 or later. See here for more details and FFmpeg's license information.