Navigation:
nginx-lua-mp4 (or simply luamp) is a LUA module for OpenResty or nginx with ngx_http_lua_module that allows on-the-fly video transcoding using ffmpeg controlled by parameters passed in the URL.
To put it simply, if you have a 1920x1080 video file
https://example.com/vids/cat.mp4
, you can request URL https://example.com/vids/w_640/cat.mp4
and you will get a 640x360 of the same video. You can crop/scale, keep aspect ratio or discard it, add padding around video (need a dynamic blurred padding? We got you covered).
leva_luamp_subs.mp4
✅ — ready, 🚧 — WIP
- General:
- ✅ Can (optionally) download original videos from the upstream, if missing from the local file system
- ✅ Configurable transcoding flags in URL, can be dropped in place of existing system without front end rewrite
- ✅ Configurable logging for debug/setup
- ✅ DPR (device pixel ratio) support
- ✅ Pass data back to nginx location after transcoding, so you can customize post-process or call more LUA modules
- ✅ Serve original on failed transcoding
- Transcoding:
- ✅ Scale/Resize
- Padding:
- ✅ Black box padding
- ✅ Blurred background padding
- ✅ Upscale protection
- 🚧 Coloured box padding
- ✅ Keep original aspect ratio
- ✅ mp4 support (output)
- 🚧 webm support (output)
- OpenResty or nginx with ngx_http_lua_module enabled
- ffmpeg 5 installed
- time utility if you have
config.logTime
enabled
Add to the http
section of the nginx config:
http {
...
lua_package_path "/absolute/path/to/nginx-lua-mp4/?.lua;;";
...
And here's minimal viable config for 4 locations you need to set up. These locations are described in the sections below:
# video location
location ~ ^/(?<luamp_flags>([0-9a-zA-Z_,\.:]+)\/|)(?<luamp_filename>[0-9a-zA-Z_\-\.]+\.mp4)$ {
# these two are required to be set regardless
set $luamp_original_video "";
set $luamp_transcoded_video "";
# these are needed to be set if you did not use them in regex matching location
set $luamp_prefix "";
set $luamp_postfix "";
#pass to transcoder location
try_files $uri @luamp_process;
}
# process/transcode location
location @luamp_process {
content_by_lua_file "/absolute/path/to/nginx-lua-mp4/nginx-lua-mp4.lua";
}
# cache location
location =/luamp-cache {
internal;
root /;
index off;
set_unescape_uri $luamp_transcoded_video $arg_luamp_cached_video_path;
try_files $luamp_transcoded_video =404;
}
# upstream location
location =/luamp-upstream {
internal;
rewrite ^(.+)$ $luamp_original_video break;
proxy_pass https://old-cdn.example.com;
}
This location used as an entry point and to set initial variables. This is usually a location with a .mp4
at the end.
There are two variables you need to set
/initialise: $luamp_original_video
and $luamp_transcoded_video
.
There are four variables that may be used as a named capture group in location regex: luamp_prefix
, luamp_flags
, luamp_postfix
, luamp_filename
.
For example:
https://example.com/asset/video/width_1980,height_1080,crop_padding/2019/12/new_year_boardgames_party.mp4
luamp_prefix: asset/video/
luamp_flags: width_1980,height_1080,crop_padding
luamp_postfix: 2019/12/
luamp_filename: new_year_boardgames_party.mp4
If you do not need prefix and postfix, you can omit them from the regexp, but do make sure you set
them to an empty string in the location. Here's the minimal viable example for simpler URLs with no prefix/postfix:
location ~ ^/(?<luamp_flags>([0-9a-zA-Z_,\.:]+)\/|)(?<luamp_filename>[0-9a-zA-Z_\-\.]+\.mp4)$ {
# these two are required to be set regardless
set $luamp_original_video "";
set $luamp_transcoded_video "";
# these are needed to be set if you did not use them in regex matching location
set $luamp_prefix "";
set $luamp_postfix "";
#pass to transcoder location
try_files $uri @luamp_process;
}
prefix
, postfix
and filename
are passed to the os.execute()
with following sanitisation:
- alphanumeric symbols, underscores, dots and slashes are allowed.
- all other symbols are stripped.
- then, double dots are stripped.
These sanitisation rules are enough to prevent shell injections and path traversal. However, especially if you decide to modify luamp code, consider using following capture patterns in the video location:
(?<luamp_prefix>[0-9a-zA-Z_\-\.\/]+\/)
(?<luamp_flags>([0-9a-zA-Z_,\.:]+)\/|)
(?<luamp_postfix>[0-9a-zA-Z_\-\.\/]+\/)
(?<luamp_filename>[0-9a-zA-Z_\-\.]+\.mp4)
Process location is pretty simple, it just passes execution to the LUA part of luamp module:
location @luamp_process {
content_by_lua_file "/absolute/path/to/nginx-lua-mp4/nginx-lua-mp4.lua";
}
Cache location is where previously transcoded videos are served from:
location =/luamp-cache {
internal;
root /;
index off;
set_unescape_uri $luamp_transcoded_video $arg_luamp_cached_video_path;
try_files $luamp_transcoded_video =404;
}
Upstream location is used when you have no original video files stored on your local file system (perhaps, those are stored on your old CDN, or elsewhere).
If luamp
finds no original file to transcode, it will attempt to download it from the upstream specified:
location =/luamp-upstream {
internal;
rewrite ^(.+)$ $luamp_original_video break;
proxy_pass https://old-cdn.example.com;
}
$luamp_original_video
is set within config.getOriginalsUpstreamPath
function that can be configured in luamp
config.lua. You can apply whatever logic you may need there to dynamically generate path for the upstream.
Go to the directory you downloaded luamp
to:
$ cd /absolute/path/to/nginx-lua-mp4/
Create a config file by copying the example config:
$ cp config.lua.example config.lua
Open it with a text editor of your choice and change the variables you feel like changing.
$ nano config.lua
When set to true
, luamp
will attempt to download missing original videos from the upstream. Set it to false
if you have original videos provided by other means to this directory:
config.mediaBaseFilepath/$prefix/$postfix/$filename
Path to the ffmpeg
executable. Can be figured out by using which
command in the terminal:
$ which ffmpeg
/usr/local/bin/ffmpeg
Where to redirect ffmpeg
output if config.logFfmpegOutput
is set to false.
For *nix (default value):
config.ffmpegDevNull = '2> /dev/null' -- nix
For win:
config.ffmpegDevNull = '2>NUL' -- win
Use this table to customize how flags are called in your URLs. Defaults are one letter flags like w
for width
, but you can customise these by editing left side of flagMap
table:
One letter flags (except for DPR) if you want to use flags like w_200,h_180,c_pad
:
['c'] = 'crop',
['b'] = 'background',
['dpr'] = 'dpr',
['h'] = 'height',
['w'] = 'width',
Full flags if you want to use flags like width_200,height_180,crop_pad
:
['crop'] = 'crop',
['background'] = 'background',
['dpr'] = 'dpr',
['height'] = 'height',
['width'] = 'width',
Customize this function to preprocess flags or their values. Return values should contain values that are present in config.flagMap
and config.flagValueMap
.
Character that is used to separate different flags in URL, e.g. commas in /w_1280,h_960,c_pad/
.
Character that is used to separate flag name from the value, e.g. underscores in /w_1280,h_960,c_pad/
.
Similar to config.flagMap
above, but for non-number flag values rather than flag names.
Default flag values, e.g. c_pad
, c_lpad
or c_fill
, also b_blurred
:
config.flagValueMap = {
['pad'] = 'padding',
['lpad'] = 'limited_padding',
['fill'] = 'fill',
['blurred'] = 'blur',
}
Full flag values, e.g. c_padding
or c_limited-padding
:
config.flagValueMap = {
['pading'] = 'padding',
['limited-padding'] = 'limited_padding',
['blurred'] = 'blur',
}
Function that is used to generate a URL for upstream. prefix
, postfix
and filename
are provided to the function and you can also use LUA ngx API.
Whether to log whole luamp
process. Useful for initial setup and for debug.
Whether to log ffmpeg
output. Note that ffmpeg
outputs to stderr
, and if logFfmpegOutput
is enabled, it will log to nginx's error.log
.
Log level, available values: ngx.STDERR
, ngx.EMERG
, ngx.ALERT
, ngx.CRIT
, ngx.ERR
, ngx.WARN
, ngx.NOTICE
, ngx.INFO
, ngx.DEBUG
.
Whether to prepend ffmpeg
command with time
utility, if you wish to log time spent in transcoding.
Limit the output video's maximum height or width. If the resulting height or width is exceeding the limit (for example, after a high DPR calculation), it will be capped at the config.maxHeight
and config.maxWidth
.
Where videos (both originals and transcoded ones) should be stored. Usually, a directory where assets are stored. Should be readable/writable for nginx.
Minimum required size (in bytes) for the transcoded file to not be considered broken and deleted (default is 1KB).
During the transcoding, errors may occur and ffmpeg sometimes leaves corrupt files on the FS. Those are usually either 0B or just a few bytes of header. Luamp will delete those that are less than minimumTranscodedFileSize
bytes.
Serve original file when transcode failed. If set to false
, luamp will respond with 404 in this case
To update luamp, just do a git pull
.
After the pull, there may be some additions to the config file. Now, one can use a diff
tool, but it will spew out all the differences, which can be overwhelming when you customize a lot in your config as compared to what is in the example config:
nginx-lua-mp4 $ diff config.lua config.lua.example
52c52
< config.logEnabled = true
---
> config.logEnabled = false
59c59
< config.logFfmpegOutput = true
---
> config.logFfmpegOutput = false
66c66,72
< config.logTime = true
---
> config.logTime = false
>
> -- top limit for output video height (default 4k UHD)
> config.maxHeight = 2160
>
> -- top limit for output video width (default 4k UHD)
> config.maxWidth = 3840
You definitely want to keep what was customized by you but also to get new config options that may be necessary. This one-liner will show you all the new config entries, omitting what was customised, given you did not move options around and just edited them in place:
nginx-lua-mp4 $ sdiff -s config.lua config.lua.example | grep -e '\s*>' | sed -ne "s/^[[:space:]]*>\t//p"
-- top limit for output video height (default 4k UHD)
config.maxHeight = 2160
-- top limit for output video width (default 4k UHD)
config.maxWidth = 3840
You can now just copy and paste these lines above the return config
in config.lua
and that's it 👌
Available values:
blurred
— when padding is enabled (withc_pad
orc_lpad
), the padding box will contain an upscaled blurred video.
Available values:
pad
— when resizing a video, aspect ratio will be preserved and padding box will be added to keep the aspect ratio.lpad
— same aspad
but original video will not be scaled up.fill
— scales and crops a video to fit specified dimensions without distortion, scaling it to fill the space and cropping any excess.
Available values: float or integer number.
Available values: integer number.
Available values: integer number.
Available values:
- integer number for pixels
- decimal number in range
(0, 1)
for percentage: 0.25 is 25% of resulting width (after DPR is applied)
Available values:
- integer number for pixels
- decimal number in range
(0, 1)
for percentage: 0.25 is 25% of resulting height (after DPR is applied)