@torrent-tv/proxy
v2.9.24
Published
Torrent proxy client that exposes webseed-like HTTP stream endpoint.
Downloads
4,527
Readme
@torrent-tv/proxy
A lightweight Node.js service that streams torrent content to browsers via a direct WebRTC P2P data channel or, when needed, HTTP. It handles torrent fetching, codec detection, and on-demand HLS transcoding with ffmpeg.
Why this exists
- Browsers cannot consume torrents directly.
- This service exposes torrent files through HTTP (
/stream) with Range support, and through a WebRTC data channel for NAT-traversed streaming. - It can transcode audio (or video + audio) to HLS on demand so the browser can always play the content regardless of codec support.
- It registers itself in an external registry server and maintains a persistent tunnel WebSocket so the server can route browser requests and WebRTC signals to it.
Architecture Overview
graph TB
subgraph Browser
WP[WebRtcProxy]
HLS[HLS.js + WebRtcHlsLoader]
end
subgraph Server["Registry Server"]
API[REST API]
TS[ProxyTunnelServer]
SH[SignalHub /ws/browser-signal]
end
subgraph Proxy["@torrent-tv/proxy (Fastify)"]
TC[TunnelClient]
WM[WebRtcManager]
HC[HealthCollector]
DCH[DataChannelHandler]
PP[PlaybackPlanner]
HLM[HlsSessionManager]
TP[TorrentPool]
FF[ffmpeg]
end
TC -->|"persistent WebSocket /ws/proxy-tunnel"| TS
TC -->|re-register on reconnect| API
HC -->|metrics: cpu, mem, load| TC
TS <-->|signal forward| SH
SH <-->|WebSocket| WP
WP <-->|"P2P data channel (STUN)"| WM
WM --> DCH
DCH --> TP
DCH --> HLM
HLM --> FF
HLS -->|segment/manifest fetches| WPService Internals
TunnelClient
Opens one persistent WebSocket to the registry server's /ws/proxy-tunnel endpoint on startup.
Reconnects automatically with back-off on unexpected close.
Handles three inbound message types from the server:
| Message type | What the proxy does |
|---|---|
| health-request | Calls HealthCollector, sends health-response back through tunnel |
| signal | Forwards SDP offer or ICE candidate to WebRtcManager |
| relay-request | Fetches the path from local Fastify, streams relay-response back |
WebRtcManager
Manages RTCPeerConnection sessions keyed by sessionId. On receiving an SDP offer from the server it creates a peer connection using node-datachannel, generates an answer, and exchanges ICE candidates through the tunnel. When the data channel opens it hands it off to DataChannelHandler.
sequenceDiagram
participant S as Server (via TunnelClient)
participant WM as WebRtcManager
participant DC as DataChannelHandler
S->>WM: { type:"offer", sessionId, sdp }
WM->>WM: createPeerConnection(sessionId)
WM->>WM: setRemoteDescription(offer)
WM->>WM: createAnswer()
WM->>S: { type:"answer", sdp }
loop ICE candidates
WM->>S: { type:"candidate", candidate, mid }
S->>WM: { type:"candidate", … }
end
Note over WM: data channel opens
WM->>DC: handleChannel(dataChannel)DataChannelHandler
Receives JSON request messages over the data channel and dispatches them to the proxy's local Fastify server. Responses are streamed back as base64 response-chunk messages.
Wire protocol (browser ↔ proxy):
| Direction | Type | Key fields |
|---|---|---|
| Browser → Proxy | request | requestId, method, path, query, headers, body |
| Proxy → Browser | response-start | requestId, status, headers |
| Proxy → Browser | response-chunk | requestId, data (base64), done: true\|false |
| Proxy → Browser | response-error | requestId, error |
| Browser → Proxy | ping | id |
| Proxy → Browser | pong | id |
HealthCollector
Collects system-level health metrics on every request from the server:
| Metric | Range | Description |
|---|---|---|
| cpuLoad | 0 – ∞ | 1-minute load average divided by CPU count (>1 = overloaded) |
| memFree | 0 – 1 | Free memory fraction |
| activeSessions | 0 – ∞ | Number of active HLS transcode sessions |
The browser uses these metrics together with tunnel RTT to score proxies:
score = memFree × 0.4 + (1 - clamp(cpuLoad, 0, 1)) × 0.4 − (rttMs / 2000) × 0.2PlaybackPlanner
Determines whether a file should be streamed directly or transcoded to HLS, and caches the result per (sourceKey, fileIndex).
Before running ffprobe it calls TorrentPool.prefetchFileEdges() which opens WebTorrent read streams for the first 256 KB and last 2 MB of the file. This forces WebTorrent to prioritise those torrent pieces, ensuring the MOOV atom (usually at the end of non-faststart MP4 files) is available before codec detection runs. The timeout is 5 minutes; if it elapses the plan still proceeds with whatever data is available and defaults to direct mode when the codec is unknown.
HlsSessionManager & ffmpeg
Creates and manages ffmpeg-based HLS transcode sessions. Sessions are keyed by sourceKey:fileIndex:mode:startPosition and shared across consumers.
Seek-to-position: createOrGetSession accepts startPositionSeconds which allows restarting a transcode from an arbitrary point. ffmpeg is invoked with -ss <N> (fast keyframe seek before -i) and -output_ts_offset <N> (shifts output PTS) so that video.currentTime in the browser reflects the original timeline position rather than resetting to zero. Start positions are rounded to 10-second buckets for session reuse.
sequenceDiagram
participant B as Browser (via DataChannel)
participant H as HlsSessionManager
participant F as ffmpeg
B->>H: POST /api/transcode-sessions (consumerId, mode)
H->>F: start (or reuse) ffmpeg process
H-->>B: { sessionId, playlistPath }
loop segment requests
B->>H: GET /transcode/:sessionId/seg000.ts
H-->>B: MPEG-TS segment (via data channel chunk)
end
B->>H: GET /api/transcode-sessions/:id/progress
H-->>B: { percent, speed, remainingSeconds, … }
B->>H: POST /api/transcode-sessions/:id/release (consumerId)
H->>H: remove consumer
alt last consumer released
H->>F: kill process + cleanup segments
endHTTP API
Base URL examples use http://127.0.0.1:9090.
Health
GET /health
GET /healthzRegister a source
POST /api/sources
Content-Type: application/json
{
"sourceType": "magnet", # or "torrent" (base64-encoded bytes)
"source": "magnet:?xt=urn:btih:…"
}Response: { "sourceKey": "…" }
Build playback plan
POST /api/playback-plan
Content-Type: application/json
{
"sourceKey": "<sourceKey>",
"fileIndex": 0,
"userAgent": "Mozilla/5.0 …"
}Response:
{
"mode": "direct",
"directUrl": "http://127.0.0.1:9090/stream?sourceKey=…&fileIndex=0",
"reason": "audio-codec-supported",
"audioCodec": "aac",
"videoCodec": "h264"
}mode is "direct" or "hls".
Direct stream
GET /stream?sourceKey=<key>&fileIndex=0Supports HTTP Range requests.
Source stats
GET /api/sources/:sourceKey/stats?fileIndex=0Returns live torrent statistics for polling during the metadata wait phase:
{
"numPeers": 4,
"downloadSpeed": 1258291,
"uploadSpeed": 0,
"fileProgress": 0.008,
"fileDownloaded": 10485760,
"fileLength": 1299000000
}fileProgress is 0–1 and reflects only the pieces that have been downloaded so far (initially the prefetched head + tail). downloadSpeed and uploadSpeed are in bytes/s.
Create HLS transcode session
POST /api/transcode-sessions
Content-Type: application/json
{
"sourceKey": "<key>",
"fileIndex": 0,
"transcodeVideo": false,
"consumerId": "uuid",
"fileName": "Episode01.mkv",
"startPositionSeconds": 300
}startPositionSeconds is optional (default 0). When set, ffmpeg fast-seeks to the specified position and shifts output timestamps accordingly, enabling seek-to-position playback.
Response: { "sessionId": "…", "playlistPath": "/transcode/<id>/index.m3u8" }
Poll transcode progress
GET /api/transcode-sessions/:sessionId/progressReturns: percent, processedSeconds, startPositionSeconds, totalSeconds, remainingSeconds, speed, warmupPercent, warmupRemainingSeconds.
Release consumer
POST /api/transcode-sessions/:sessionId/release
Content-Type: application/json
{ "consumerId": "uuid", "reason": "pagehide" }When the last consumer is released, the transcode session stops and temp files are cleaned up.
Requirements
- Node.js 18+ (ESM, built-in
fetch). - ffmpeg is required only when transcoding is enabled (bundled via
ffmpeg-staticby default).
Run
npm install
npm start -- --server-url http://localhost:3000CLI Options
| Option | Default | Description |
|--------|---------|-------------|
| --server-url | — | (Required) Base URL of the registry server |
| --host | 127.0.0.1 | Bind host |
| --port | 9090 | Preferred local port (auto-increments if taken) |
| --public-base-url | — | Externally reachable base URL advertised to registry |
| --id | auto | Stable proxy client ID |
| --name | hostname | Display name in registry |
| --token | — | Auth token for register/heartbeat |
| --ffmpeg-bin | bundled | Path to custom ffmpeg binary |
| --no-transcode-audio | — | Disable HLS audio transcoding |
| --help | — | Print all options and exit |
Docker
docker build -t torrent-tv-proxy .
docker run torrent-tv-proxy --server-url http://my-server:8080Full End-to-End Flow
sequenceDiagram
participant B as Browser
participant S as Registry Server
participant P as Proxy
Note over P,S: Startup
P->>S: POST /api/proxy-clients/register
P->>S: WebSocket /ws/proxy-tunnel (persistent)
Note over B,P: Playback start
B->>S: GET /api/proxy-clients/health
S->>P: health-request via tunnel
P-->>S: health-response (cpu, mem, activeSessions)
S-->>B: scored proxy list
Note over B,P: WebRTC setup
B->>S: WebSocket /ws/browser-signal
B->>B: RTCPeerConnection + DataChannel + createOffer
B->>S: { type:"offer", proxyId, sdp }
S->>P: forward via tunnel
P->>P: createAnswer
P->>S: { type:"answer", sdp }
S->>B: forward
Note over B,P: ICE candidates exchanged same way
Note over B,P: Data channel opens (P2P, STUN-assisted)
Note over B,P: Streaming
B->>P: request via data channel: POST /api/sources
P-->>B: response-chunk: { sourceKey }
B->>P: request via data channel: POST /api/playback-plan
P-->>B: response-chunk: { mode, audioCodec, videoCodec, … }
B->>P: request via data channel: POST /api/transcode-sessions
P-->>B: response-chunk: { sessionId, playlistPath }
B->>P: HLS.js fetches: GET /transcode/:id/index.m3u8
B->>P: HLS.js fetches: GET /transcode/:id/seg000.ts …
Note over B,P: All via data channel — no server relayNotes
- HLS session temp files are in the OS temp directory and cleaned up automatically.
- Transcode sessions are cached by
sourceKey:fileIndex:mode:startPositionand shared across consumers. - The source registry is in-memory and bounded (old entries evicted).
- The proxy reconnects to the server automatically on tunnel disconnect.
prefetchFileEdgesdownloads only the file's head and tail (≈ 2.3 MB total), not the full file, before codec detection runs.- When
startPositionSeconds > 0, ffmpeg uses fast seek (-ssbefore-i) — it starts from the nearest keyframe at or before the requested position. The first few frames may be slightly before the exact seek time.
License
GPL-3.0-or-later (see LICENSE). Third-party dependencies keep their own licenses.
Bundled ffmpeg binaries (ffmpeg-static) are GPL-compatible.
