Generate .srt subtitle files next to video files using local MLX Whisper and LLM translation
Project description
add-srt-next-to-videos
Generate .srt subtitle files next to video files. Point it at a file or folder, get subtitles.
Runs entirely on your Mac — MLX Whisper handles speech-to-text, a local LLM handles translation. Nothing leaves your machine.
Requirements
- Apple Silicon Mac (M1/M2/M3/M4) — MLX only runs on Apple Silicon
- Python 3.10+
- pipx for system-wide install
- For translation: any local LLM server with an OpenAI-compatible
/v1/chat/completionsendpoint — LM Studio, Ollama, etc.
Install
# Homebrew
brew install yuyangchee98/tap/add-srt-next-to-videos
# From PyPI
pip install add-srt-next-to-videos
# Or with pipx (isolated install, recommended)
pipx install add-srt-next-to-videos
# Or directly from GitHub
pipx install git+https://github.com/yuyangchee98/add-srt-next-to-videos.git
After install, create a .env file in the directory where you run the command (or your home directory):
cp .env.example .env
# edit .env with your model preferences
To uninstall: pipx uninstall add-srt-next-to-videos
For development:
git clone https://github.com/yuyangchee98/add-srt-next-to-videos.git
cd add-srt-next-to-videos
uv venv && uv pip install -e .
Configuration
Copy .env.example to .env and edit:
# Whisper model for speech-to-text (HuggingFace repo)
WHISPER_MODEL=mlx-community/whisper-large-v3-mlx
# LLM for translation (any server with an OpenAI-compatible /v1/chat/completions endpoint)
# Works with: LM Studio, Ollama, vLLM, llama.cpp server, etc.
LLM_URL=http://localhost:1234
LLM_MODEL=qwen3-32b-mlx
Whisper model weights (~3GB) download automatically on first run and cache in ~/.cache/huggingface/.
The LLM is only used for translation. If you only need transcription (source language subs), you don't need an LLM server running.
Usage
# Transcribe Japanese audio -> Japanese subtitles
add-srt-next-to-videos ./videos -l ja --subs ja
# Translate to English
add-srt-next-to-videos ./videos -l ja --subs en
# Multiple languages at once
add-srt-next-to-videos ./videos -l ja --subs ja,en,zh
# Dual-language subs (both languages in one file, for language learning)
add-srt-next-to-videos ./videos -l ja --subs ja-en
# Mix and match
add-srt-next-to-videos ./videos -l ja --subs ja,en,ja-en,ja-zh
# Single file
add-srt-next-to-videos ./episode01.mp4 -l ja --subs ja,en
Works on single files or entire directories (recursive). Existing .srt files are skipped unless you pass --overwrite.
Output files
The sub spec determines the filename:
video.mp4 -> video.ja.srt (transcription)
video.mp4 -> video.en.srt (translation)
video.mp4 -> video.ja-en.srt (dual: japanese on top, english below)
A dual-language .ja-en.srt looks like this:
1
00:00:05,000 --> 00:00:10,679
母ちゃん母ちゃん今日はどこ行くの
Mom, mom, where are we going today?
2
00:00:10,679 --> 00:00:12,060
美容院よ
The beauty salon.
How it works
- Whisper transcribes the audio once per video — timestamped segments in the source language
- LLM translates each segment's text to other languages (only if needed)
Whisper handles transcription only. All translation goes through your local LLM for better accuracy.
The tool transcribes once and caches the result per video, so --subs ja,en,ja-en runs Whisper once and translates to English once, even though two outputs use the English translation.
CLI reference
add-srt-next-to-videos [-h] [--language LANGUAGE] [--subs SUBS] [--overwrite] path
| Flag | Short | Default | What it does |
|---|---|---|---|
path |
(required) | Video file or directory (recursive) | |
--language |
-l |
auto-detect | Source language (ja, en, zh, etc.) |
--subs |
-s |
source lang | What to generate: ja, en, ja-en, or comma-separated |
--overwrite |
off | Regenerate existing .srt files |
Whisper model and LLM settings are configured in .env (see Configuration).
Supported video formats
.mp4 .mkv .avi .mov .webm .flv .wmv .m4v .mpg .mpeg .ts .vob
Notes
-
If a run gets interrupted, just re-run the same command — it skips videos that already have their
.srtfiles -
Language codes are ISO 639-1. Whisper supports these languages for transcription:
afAfrikaans,arArabic,hyArmenian,azAzerbaijani,beBelarusian,bsBosnian,bgBulgarian,caCatalan,zhChinese,hrCroatian,csCzech,daDanish,nlDutch,enEnglish,etEstonian,fiFinnish,frFrench,glGalician,deGerman,elGreek,heHebrew,hiHindi,huHungarian,isIcelandic,idIndonesian,itItalian,jaJapanese,knKannada,kkKazakh,koKorean,lvLatvian,ltLithuanian,mkMacedonian,msMalay,mrMarathi,miMaori,neNepali,noNorwegian,faPersian,plPolish,ptPortuguese,roRomanian,ruRussian,srSerbian,skSlovak,slSlovenian,esSpanish,swSwahili,svSwedish,tlTagalog,taTamil,thThai,trTurkish,ukUkrainian,urUrdu,viVietnamese,cyWelshTranslation to any language depends on your LLM's capabilities
License
MIT
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file add_srt_next_to_videos-0.1.1.tar.gz.
File metadata
- Download URL: add_srt_next_to_videos-0.1.1.tar.gz
- Upload date:
- Size: 8.9 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.15
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
9851f69a04ac51c1a647326b24e786a104e7a84b3375a3c739a8374fa875c33d
|
|
| MD5 |
7f1744dd26da7d86f3311f71a8262093
|
|
| BLAKE2b-256 |
d29cd315bc586ecdf65dc2c39bc1232470820d5484207bfd1872fe6e9d207c42
|
File details
Details for the file add_srt_next_to_videos-0.1.1-py3-none-any.whl.
File metadata
- Download URL: add_srt_next_to_videos-0.1.1-py3-none-any.whl
- Upload date:
- Size: 9.7 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.11.15
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
3e1f012e9fdd84923f2a514b81c6c6d10ea9214624a3414d0f2b828ded0f4175
|
|
| MD5 |
09acff671d72f1a341178705e431a151
|
|
| BLAKE2b-256 |
c1352c4a7ebc84cb8f4b35aa1b7ce818ddf604c31e07f3dc473e3c95e3ab7dc7
|