2020-09-27 11:14:22 +02:00
|
|
|
package srt
|
|
|
|
|
|
|
|
import (
|
2020-09-29 20:56:31 +02:00
|
|
|
"bufio"
|
|
|
|
"fmt"
|
2020-09-27 11:14:22 +02:00
|
|
|
"log"
|
2020-09-29 16:49:30 +02:00
|
|
|
"net"
|
2020-09-29 20:56:31 +02:00
|
|
|
"os/exec"
|
2020-09-29 16:49:30 +02:00
|
|
|
"strconv"
|
2020-09-27 11:14:22 +02:00
|
|
|
|
2020-09-27 20:43:00 +02:00
|
|
|
"github.com/haivision/srtgo"
|
2020-09-28 16:32:35 +02:00
|
|
|
"github.com/pion/rtp"
|
2020-09-27 11:14:22 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// Options holds web package configuration
|
|
|
|
type Options struct {
|
|
|
|
ListenAddress string
|
2020-09-29 16:49:30 +02:00
|
|
|
MaxClients int
|
|
|
|
}
|
|
|
|
|
|
|
|
// Split host and port from listen address
|
|
|
|
func splitHostPort(hostport string) (string, uint16) {
|
|
|
|
host, portS, err := net.SplitHostPort(hostport)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Failed to split host and port from %s", hostport)
|
|
|
|
}
|
|
|
|
if host == "" {
|
|
|
|
host = "0.0.0.0"
|
|
|
|
}
|
|
|
|
port64, err := strconv.ParseUint(portS, 10, 16)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Port is not a integer: %s", err)
|
|
|
|
}
|
|
|
|
return host, uint16(port64)
|
2020-09-27 11:14:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Serve SRT server
|
|
|
|
func Serve(cfg *Options) {
|
2020-09-27 20:43:00 +02:00
|
|
|
options := make(map[string]string)
|
2020-09-28 16:32:35 +02:00
|
|
|
options["transtype"] = "live"
|
2020-09-27 20:43:00 +02:00
|
|
|
|
2020-09-29 16:49:30 +02:00
|
|
|
log.Printf("SRT server listening on %s", cfg.ListenAddress)
|
|
|
|
host, port := splitHostPort(cfg.ListenAddress)
|
|
|
|
sck := srtgo.NewSrtSocket(host, uint16(port), options)
|
|
|
|
sck.Listen(cfg.MaxClients)
|
2020-09-27 11:14:22 +02:00
|
|
|
|
2020-09-27 20:43:00 +02:00
|
|
|
for {
|
2020-09-27 22:10:47 +02:00
|
|
|
s, err := sck.Accept()
|
|
|
|
if err != nil {
|
2020-09-28 16:32:35 +02:00
|
|
|
log.Println("Error occurred while accepting request:", err)
|
2020-09-27 22:27:18 +02:00
|
|
|
continue
|
2020-09-27 20:43:00 +02:00
|
|
|
}
|
2020-09-27 22:10:47 +02:00
|
|
|
|
2020-09-29 20:56:31 +02:00
|
|
|
// Launch ffmpeg to stream on other RTMP servers
|
|
|
|
ffmpeg := exec.Command("ffmpeg", "-re", "-i", "pipe:0", "-f", "flv", "-c:v", "libx264", "-preset", "veryfast", "-maxrate", "3000k", "-bufsize", "6000k", "-pix_fmt", "yuv420p", "-g", "50", "-c:a", "aac", "-b:a", "160k", "-ac", "2", "-ar", "44100", fmt.Sprintf("rtmp://live.twitch.tv/app/%s", "TWITCH_STREAM_KEY")) //nolint
|
|
|
|
ffmpegIn, _ := ffmpeg.StdinPipe()
|
|
|
|
ffmpegOut, _ := ffmpeg.StderrPipe()
|
|
|
|
|
2020-09-28 16:32:35 +02:00
|
|
|
buff := make([]byte, 2048)
|
|
|
|
n, err := s.Read(buff, 10000)
|
2020-09-29 20:56:31 +02:00
|
|
|
ffmpegIn.Write(buff[:n])
|
2020-09-28 16:32:35 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Println("Error occurred while reading SRT socket:", err)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if n == 0 {
|
|
|
|
// End of stream
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unmarshal the incoming packet
|
|
|
|
packet := &rtp.Packet{}
|
|
|
|
if err = packet.Unmarshal(buff[:n]); err != nil {
|
|
|
|
log.Println("Error occured while unmarshaling SRT:", err)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// videoTrack, err := peerConnection.NewTrack(payloadType, packet.SSRC, "video", "pion")
|
|
|
|
|
2020-09-29 20:56:31 +02:00
|
|
|
if err := ffmpeg.Start(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Log ffmpeg output
|
|
|
|
go func() {
|
|
|
|
scanner := bufio.NewScanner(ffmpegOut)
|
|
|
|
for scanner.Scan() {
|
|
|
|
log.Println(scanner.Text())
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2020-09-28 16:32:35 +02:00
|
|
|
// Read RTP packets forever and send them to the WebRTC Client
|
|
|
|
for {
|
|
|
|
n, err := s.Read(buff, 10000)
|
|
|
|
if err != nil {
|
|
|
|
log.Println("Error occured while reading SRT socket:", err)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Printf("Received %d bytes", n)
|
|
|
|
|
|
|
|
packet := &rtp.Packet{}
|
2020-09-29 20:56:31 +02:00
|
|
|
ffmpegIn.Write(buff[:n])
|
2020-09-28 16:32:35 +02:00
|
|
|
if err := packet.Unmarshal(buff[:n]); err != nil {
|
|
|
|
panic(err)
|
2020-09-27 22:10:47 +02:00
|
|
|
}
|
2020-09-28 16:32:35 +02:00
|
|
|
payloadType := uint8(22) // FIXME put vp8 payload
|
|
|
|
packet.Header.PayloadType = payloadType
|
|
|
|
|
|
|
|
//err := videoTrack.WriteRTP(packet)
|
|
|
|
}
|
2020-09-27 11:14:22 +02:00
|
|
|
}
|
|
|
|
}
|