Skip to content

Latest commit

 

History

History
221 lines (153 loc) · 10.4 KB

README.md

File metadata and controls

221 lines (153 loc) · 10.4 KB

RTMP Live

What is this?

This repository provides a comprehensive guide and code samples for creating a small streaming platform based on the RTMP (Real-Time Messaging Protocol). The platform enables live streaming capabilities and leverages NGINX RTMP for receiving video streams. Additionally, the repository includes functionality to play the recorded videos via HTTP directly from another pool of servers.

Additionally, a service discovery process is included to report the active streams to an API. The API, integrated with Redis, returns the server and manifest path required for playback.

graph LR
    A[Edge] -- Which server should I request video? --> B[API]
    B -- Get server --> C[Redis]
    B -- Response with Origin A --> A
    A -- Request content --> D[Origin A]
    E[Origin B]
Loading

Platform components:

  • Origin: ingest, storage and content origin
  • Edge: CDN, server you use to play the video
  • API: tracks Origin servers

What's the stack behind it?

This small live streaming platform relies on the following projects:

  • NGINX-RTMP - the widely, battle-tested and probably the most famous RTMP server
  • NGINX - the most used werb server in the world
  • Lua - a simple yet very powerful programing language 🇧🇷
  • Go - a good language to build HTTP APIs, workers, daemons and every kind of distribued system service

How to use

There are some requirements you need to run this project:

Now you are good to go!

To use the platform, follow these steps:

  1. Open your terminal and execute the command: docker compose up
  2. Once all the components are up and running, launch OBS Studio on your computer.
  3. Configure OBS Studio to stream via RTMP using the following settings:
    Stream Type: Custom Streaming Server

    URL: rtmp://localhost:1935/stream

    Stream Key: golive
  1. Start your live streaming session in OBS Studio. The platform will now receive your live stream and make it available for playback.
  2. Use a player like VLC and point it to http://127.0.0.1:8080/golive/index.m3u8. You can also use a browser with a proper extension to play HLS.

There is also a test video that can be generated using ffmpeg:

ffmpeg -re -f lavfi -i "smptehdbars=rate=60:size=1920x1080" \
    -f lavfi -i "sine=frequency=1000:sample_rate=48000" \
    -vf drawtext="text='RTMP Live %{localtime\:%X}':rate=60:x=(w-tw)/2:y=(h-lh)/2:fontsize=48:fontcolor=white:box=1:boxcolor=black" \
    -f flv -c:v h264 -profile:v baseline -pix_fmt yuv420p -preset ultrafast -tune zerolatency -crf 28 -g 120 -c:a aac \
    "rtmp://localhost:1935/stream/golive"

RTMP Settings

For detailed guidance on using OBS Studio, there are plenty of tutorials available on the internet. They provide comprehensive instructions and helpful tips for a smooth streaming setup.

If everything goes fine, you will watch a colorbar video just like this:

colorbar

Edge - CDN

The Edge server, often referred to as "the frontend server" is an essential component of the Content Delivery Network (CDN). It plays a crucial role in the media streaming platform, facilitating a seamless viewing experience for users.

It is the server delivered by the platform you are using to watch the video, this is the server your media player will use to play the video.

The Edge server serves as the intermediary between the end-users and the video content they wish to watch. When you access a video on the platform, your media player interacts with the Edge server, which efficiently delivers the video content to your device. This playable URL comes through an HTTP API and it is out of the scope of this educational project.

Our Edge component here is responsible for retrieving from an HTTP API which Origin server the content will come from, and stick to it through the playback.

graph LR

style Edge fill:#FFC107, stroke:#000, stroke-width:2px, r:10px
style API fill:#4CAF50, stroke:#000, stroke-width:2px, r:10px
style Origin fill:#2196F3, stroke:#000, stroke-width:2px, r:10px

Edge("Edge") -- Which servers holds the content? --> API("HTTP API")
API -- Returns JSON with data --> Edge
Edge -- Proxies Request --> Origin("Origin Server")
Loading

A typical response from the HTTP API looks like this:

{
    "name": "golive",
    "manifest": "index.m3u8",
    "host": "127.0.0.1"
}

We use these values in the proxy_pass directive to proxy the request to the correct origin server.

location ~ "/(?<stream>[^/]*)/index.m3u8$" {
    set $target "";

    access_by_lua_block {
        ngx.var.target = router.fetch_streams(ngx.var.stream)
    }

    proxy_pass http://$target;
}

Origin

The Origin is the component responsible for receiving the video (ingest), storing and serving the original video content to the Edge servers and users.

Keys characteristics of the Origin service are:

  • Ingest: it receives the video feed from an encoder, such as Elemental or OBS Studio, serving as the entry point for content upload.
  • Packager: the Origin service packages the video for user consumption, fragments it into segments, and generates HLS manifests.
  • Storage: in addition to packaging, the Origin service stores all the video content.
  • Delivery: as the backbone of content distribution, it acts as an upstream to the Edge servers, efficiently delivering content when requested.
graph TD

style Encoder fill:#B2DFDB, stroke:#000, stroke-width:2px, r:10px
style RTMPLB fill:#FFCC80, stroke:#000, stroke-width:2px, r:10px
style OriginA fill:#BBDEFB, stroke:#000, stroke-width:2px, r:10px
style OriginB fill:#BBDEFB, stroke:#000, stroke-width:2px, r:10px

Encoder("Encoder (e.g., OBS)") --> RTMPLB("RTMP Load Balancer")
RTMPLB --> OriginA("Origin A")
RTMPLB --> OriginB("Origin B")
Loading

In live streaming, using distributed storage is impractical due to latency issues. To maintain low latency and avoid buffering during streaming, video packagers opt for local storage. This approach reduces the time it takes to access and deliver content to viewers, ensuring a greater playback experience.

To determine which server hosts specific content, such as World Cup Finals or Playing Final Fantasy, we implement a Discovery program. This program tracks the locations of various streams and provides the necessary information for efficient content delivery. By leveraging the Discovery program, our platform optimizes content distribution, guaranteeing a seamless live streaming experience for viewers.

Discovery

The Discovery service is responsible for tracking and identifying which server holds a specific streaming content. This becomes especially important when multiple encoders are feeding the Origin Service with different content, and the platform needs to determine the appropriate server(s) to deliver the content when requested by users.

It is deployed aside with the Origin service. The reason is because we continuously need to know whether the video feeding is up and running.

sequenceDiagram
    participant DS as Discovery Service
    participant FS as Filesystem
    participant API as HTTP API

    loop Watch filesystem events
        DS->>FS: Check if manifests are being created/updated
    end

    Note right of DS: Accesses the filesystem to verify if the streaming is working
    DS->>API: Report Host (IP), manifest path, stream name (e.g golive)
    Note right of DS: Sends relevant information to the HTTP API
Loading

Our RTMP server supports authorization through an on_publish callback. This functionality plays a vital role in our platform, as it allows us to ensure secure ingest of live streams. When a new stream is published, the RTMP server triggers the on_publish callback, and our platform calls the HTTP API to authorize the ingest.

application stream {
    live on;
    record off;
    notify_method get;
    on_publish http://api:9090/authorize;
}

Content delivey is supported using a location with the alias directive:

location / {
    alias /opt/data/hls/;
    types {
        application/vnd.apple.mpegurl m3u8;
        video/mp2t ts;
    }
    add_header Cache-Control no-cache;
    add_header Access-Control-Allow-Origin *;
}

API

The HTTP API used by the Discovery Service performs two critical functions. Firstly, it enables real-time updates to Redis keys, ensuring the tracking of changes in streaming manifests. By utilizing TTL for Redis keys, the API automatically removes keys when the encoder goes offline or when the live streaming session ends. As a result, the platform stops offering the corresponding live content.

You can try the API using the VSCode Rest Client extension. Open the api.http file

The API has three routes:

Your turn

A basic architecture has been described. And now it is your time to think about next steps for our live streaming platform:

  • Best possible experience: to ensure the best possible viewer experience, explore implementing adaptive bitrate streaming. Read about Adaptive bitrate streaming
  • Increased Resiliency: what happens if the HTTP API goes offline for 5 minutes? How can the system handle and recover from such scenarios without compromising content availability?
  • Scalability: to reduce latency while maintaining content delivery efficiency, explore techniques that can lower latency without reducing the segment size

This documentation was heavily inspired by @leandromoreira CDN up and running