video_press/README.md

5.8 KiB

VideoPress — FFmpeg Video Compressor

NOTE: This application was built with the help of AI.

  • The application code has been reviewed by this developer.
  • The application code has been tested by this developer.

If you find any issues, please feel free to post an issue, and / or a pull request for a fix.

What is VideoPress?

A web-based video compression tool served via Gunicorn + gevent (WSGI), containerised with Docker. FFmpeg compresses video files to approximately 1/3 their original size. All file-system access is restricted to a single configurable media root for security.

  1. Set your folder to scan, and minimum size to scan for.

Image 1

  1. Select from the files found.

Image 2

  1. Start the Compression and See the Progress

Image 3

  1. Get an email notification when a compresion run finishes.

Image 4

Image 5


Step. 1. You have two options

A. Build the image

docker build -t videopress .

B. Use the pre-built image

Already included in the docker-compose.yml file witht his project.

Step 2. Run — replace /your/video/path with the real path on your host

If you want to persist data between updates, changes, etc.; then you need to create a "data" directory in the same location you store your 'docker-compose.yml' file.

mkdir -p data

Ensure the 'data' directory is accessible with permissions = 775.

chmod -R 775 ./data

A. Using docker run:

  • with an image you build:
docker run -d \
  --name videopress \
  --restart unless-stopped \
  -p 8080:8080 \
  -v /your/video/path:/media \
  -v ./data:/data \
  videopress
  • with the pre-built image:
docker run -d \
--name videopress \
--restart unless-stopped \
-p 8080:8080 \
-v /your/video/path:/media \
-v ./data:/data \
bmcgonag/videopress:latest

B. Using Docker Compose

docker compose up -d

Step 3. Open http://localhost:8080


Bare-metal (without Docker)

Requirements

Tool Version
Python 3.8+
FFmpeg + FFprobe any recent
pip packages see requirements.txt
# Create a python virtual environment to run in
python3 -m venv videopress

#start the virtual environment
source ./videopress/bin/activate

# Install Python dependencies
pip install -r requirements.txt

# Development server (Flask built-in — not for production)
./start.sh

# Production server (Gunicorn + gevent)
MEDIA_ROOT=/your/video/path ./start.sh --prod
# or on a custom port:
MEDIA_ROOT=/your/video/path ./start.sh --prod 9000

Security model

Every API call that accepts a path validates it with safe_path() before any OS operation. safe_path() resolves symlinks and asserts the result is inside MEDIA_ROOT. Requests that attempt directory traversal are rejected with HTTP 403. The container runs as a non-root user (e.g. your id. You can find this by entering the command id in the terminal, make note of the number for uid - usually 1000).


Architecture

Browser  ──HTTP──▶  Gunicorn (gevent worker)
                        │
                        ├─ GET  /                 → index.html
                        ├─ GET  /api/config        → {"media_root": ...}
                        ├─ GET  /api/browse        → directory listing
                        ├─ POST /api/scan          → ffprobe metadata
                        ├─ POST /api/compress/start → launch ffmpeg thread
                        ├─ GET  /api/compress/progress/<id>  ← SSE stream
                        └─ POST /api/compress/cancel/<id>

Why gevent? SSE (/api/compress/progress) is a long-lived streaming response. Standard Gunicorn sync workers block for its entire duration. Gevent workers use cooperative greenlets so a single worker process can handle many concurrent SSE streams and normal requests simultaneously.

Why workers=1? Job state lives in an in-process Python dict. Multiple worker processes would not share it. One gevent worker with many greenlets is sufficient for this workload. To scale to multiple workers, replace the in-process job store with Redis.


Environment variables

Variable Default Description
MEDIA_ROOT /media Root directory the app may access
PORT 8080 Port Gunicorn listens on
LOG_LEVEL info Gunicorn log level

File layout

videocompressor/
├── app
    ├── __init__.py
    ├── config.py
    ├── db.py
    ├── jobs.py
    ├── media.py
    ├── notify.py
    └── routes.py
├── wsgi.py
├── gunicorn.conf.py
├── requirements.txt
├── Dockerfile
├── docker-compose.yml
├── start.sh
├── README.md
├── templates/
│   └── index.html
└── static/
    ├── css
        └── main.css
    └── js
        ├── app.js
        └── modules
            ├── browser.js
            ├── compress.js
            ├── progress.js
            ├── scan.js
            ├── session.js
            ├── settings.js
            ├── state.js
            ├── stream.js
            ├── theme.js
            └── utils.js

Contribute

Feel free to clone the repository, make updates, and submit a pull request to make this more feature rich.

Keep in mind, I like to keep things fairly simple to use.

If you need to know too much about ffmpeg and how to configure the perfect compression, then that may be too much for this app, but feel free to fork this repository and make your own as well.