stream-server/rtmp/flv/writer.go

115 lines
3.3 KiB
Go

package flv
import (
"io"
"os/exec"
"encoding/binary"
"os"
)
type FLVWriter struct {
w io.WriteCloser // abstraction as a wrapper to the actual transcoder command
}
func NewFLVWriter(stream_key string) (*FLVWriter, error) {
base_dir, _ := os.UserHomeDir()
writer := new(FLVWriter)
// spawn ffmpeg as a transcoder + hls segmenter
// most are generic commands, added a prefix to each segment url to make serving it over http easier
transcoder := exec.Command(
"ffmpeg",
"-probesize", "500",
"-i", "pipe:0",
"-c:a", "aac",
"-c:v", "h264",
"-b:v", "1M",
"-g", "30",
"-hls_time", "6",
"-hls_list_size", "4",
"-hls_base_url", "/vid/" + stream_key + "/",
"-hls_segment_type", "fmp4",
"-hls_flags", "delete_segments",
"-hls_flags", "+program_date_time",
"stream.m3u8",
)
transcoder.Dir = base_dir + "/live/" + stream_key + "/" // shift to the appropriate dir for the given stream key
flvpipe, err := transcoder.StdinPipe() // give control over the ffmpeg input as a stdin pipe (will push in flv packets as they come)
transcoder.Start() // spawn ffmpeg
if err != nil {
return nil, err
}
writer.w = flvpipe
if err = writer.write_flv_header(); err != nil { // init its header now, might as well
return nil, err
}
return writer, nil
}
func (writer *FLVWriter) Close() (error) {
return writer.w.Close()
}
// first data tag, AMF0 message but FLV spec requires the encoded data
// just requires skipping the RTMP specific bits, then just writing as is
func (writer *FLVWriter) WriteMetadataTag(data *[]byte) (err error) {
uint24_buf := make([]byte, 4)
tag_header := make([]byte, 11)
tag_header[0] = 18 // RTMP msg_type 18
binary.BigEndian.PutUint32(uint24_buf, uint32(len((*data)[16:]))) // "@setDataFrame", null, actual data
copy(tag_header[1:4], uint24_buf[1:])
if _, err = writer.w.Write(tag_header); err != nil {
return
}
if _, err = writer.w.Write((*data)[16:]); err != nil {
return
}
tag_len_buf := make([]byte, 4)
binary.BigEndian.PutUint32(tag_len_buf, uint32(len((*data)[16:]) + 11))
_, err = writer.w.Write(tag_len_buf)
return
}
// write the actual audio + video data, same as metadata except no skipping
func (writer *FLVWriter) WriteMediaTag(data *[]byte, timestamp uint32, media_type uint8) (err error) {
uint24_buf := make([]byte, 4)
tag_header := make([]byte, 11)
tag_header[0] = media_type
data_len := uint32(len(*data))
binary.BigEndian.PutUint32(uint24_buf, data_len)
copy(tag_header[1:4], uint24_buf[1:])
binary.BigEndian.PutUint32(uint24_buf, timestamp)
copy(tag_header[4:7], uint24_buf[1:])
tag_header[7] = uint24_buf[0]
if _, err = writer.w.Write(tag_header); err != nil {
return
}
if _, err = writer.w.Write(*data); err != nil {
return
}
tag_len_buf := make([]byte, 4)
binary.BigEndian.PutUint32(tag_len_buf, data_len + 11)
_, err = writer.w.Write(tag_len_buf)
return
}
// FLV file header always 9 bytes + 4 bytes for first ghost data tag
func (writer *FLVWriter) write_flv_header() (err error) {
header := make([]byte, 13)
copy(header[:3], "FLV")
header[3] = 1 // FLV version, only 1 is ever used
header[4] = 5 // 1 and 4 -> audio OR video | 5 -> audio + video
header[8] = 9 // len of header
// flv spec requires each tag + tag_len. defines a ghost tag at start with len 0
// likely for player seeking?
_, err = writer.w.Write(header)
return
}