Stream your heart's content with HLS.
I've got a CCTV camera from AliExpress, I know I can use ffmpeg hls demuxer to split up the stream into HLS segments and serve nginx over the files to make an HLS stream.
But then I learned that my camera can only do H265 stream, while HLS supports H265, no mainstream browsers support H265 natively. FFMPEG tries to transcode the video on the fly and it is a very expensive operation, even with hardware support: 400% software vs 100% hardware CPU usage. I've come up with this idea that only when someone accesses the stream should the transcoding start.
- Written in Rust, minimal resource usage: the app itself takes <5K memory to run1.
- FFMPEG/transcoding only starts first time accessing the stream.
- FFMPEG/transcoding stops when access stops.
- Comes with a simple web page: it's ready to view the result!
For an onvif compatible camera, we can use its rtsp stream.
$ docker run -d \
-e FFMPEG_INPUT="-rtsp_transport tcp -i rtsp://NAME:PASSWORD@CAMERA_IP/onvif2"
ghcr.io/simophin/hls-streamer
Refer to the rest of the documentation for the usage of FFMPEG_INPUT
.
docker-compose.yaml
services:
webcam:
image: ghcr.io/simophin/hls-streamer:latest
restart: always
ports:
- 8989:8989
devices:
# Device mapping only needed for hardware acceleration
- /dev/dri/renderD128:/dev/dri/renderD128
volumes:
# Optional. Use a tmpfs for data storage to reduce writes to your disks.
- hls-data:/data
environment:
- FFMPEG_INPUT=-vaapi_device /dev/dri/renderD128 -rtsp_transport tcp -i rtsp://USER:PASS@CAMERA_IP/onvif2 -vf format=nv12,hwupload -b:v 2M -c:v h264_vaapi
volumes:
hls-data:
driver: local
driver_opts:
type: tmpfs
device: tmpfs
o: size=200m
You'll need to set up Rust toolchain first, see rustup
Clone the source and build
$ cargo build --release
Once built, you can run via:
$ FFMPEG_INPUT=xxx ./target/release/hls-streamer
You'll also need to have ffmpeg>4.0 installed to use the app.
Only Linux system is tested at this time.
Name | Default value | Description |
---|---|---|
FFMPEG_INPUT | FFMPEG input parameters. e.g. -rtsp_transport tcp -i rtsp://NAME:PASSWORD@CAMERA_IP/onvif2 . See below for examples. |
|
HLS_DIR | /data | HLS data storage directory. It's recommended to use a memory based system like tmpfs to avoid frequent write to disks. |
LISTEN_ADDRESS | 0.0.0.0 | Http server listening address |
LISTEN_PORT | 8989 | Http server listening port |
TIMEOUT_SECONDS | 120 | The waiting time before a stream is considered idle. |
Once you have the app running, by default, you will have these two links:
http://localhost:8989
-> a simple webpage showing the HLS stream
http://localhost:8989/master.m3u8
-> the HLS playlist itself
The playlist file request will be withheld until the playlist file is generated. This is to avoid the first time you access the playlist because ffmpeg is not ready and you'd have got a 404 for that file.
These examples demostrate what you can put into FFMPEG_INPUT
environment variable.
-rtsp_transport tcp -i rtsp://NAME:PASSWORD@CAMERA_IP/onvif2 -vf copy
-vaapi_device /dev/dri/renderD128 -rtsp_transport tcp -i rtsp://NAME:PASSWORD@CAMERA_IP/onvif2 -vf format=nv12,hwupload -b:v 2M -c:v h264_vaapi
Footnotes
-
The measurement was indicative only. FFMPEG process is excluded. ↩