mptv3/main.go
2025-08-24 07:52:56 +03:00

247 lines
4.8 KiB
Go

package main
import (
"encoding/json"
"flag"
"fmt"
"io"
"net"
"net/http"
"os"
"os/exec"
"regexp"
"strings"
"time"
"golang.org/x/net/html"
)
type MPVRequest struct {
Command []string `json:"command"`
}
type MPVResponsePlayback struct {
Data float64 `json:"data"`
}
type MPV struct {
cmd *exec.Cmd
socketPath string
}
func NewMPV(streamUrl string, socketPath string) MPV {
return MPV{
cmd: exec.Command("mpv",
"--vo=gpu",
"--hwdec=vaapi",
fmt.Sprintf("--input-ipc-server=%s", socketPath),
streamUrl),
socketPath: socketPath,
}
}
func (mpv *MPV) Spawn() error {
mpv.cmd.Stdout = io.Discard
mpv.cmd.Stderr = os.Stderr
return mpv.cmd.Start()
}
func (mpv *MPV) Stop() {
mpv.cmd.Process.Kill()
mpv.cmd.Process.Wait()
}
func (mpv *MPV) ExecuteIPC(req *MPVRequest) ([]byte, error) {
reqBytes, err := json.Marshal(req)
if err != nil {
return nil, err
}
reqBytes = append(reqBytes, byte('\n'))
conn, err := net.Dial("unix", mpv.socketPath)
if err != nil {
return nil, err
}
defer conn.Close()
_, err = conn.Write(reqBytes)
if err != nil {
return nil, err
}
resBytes := make([]byte, 1024)
n, err := conn.Read(resBytes)
if err != nil {
return nil, err
}
return resBytes[:n], nil
}
func (mpv *MPV) InquirePlayback() (float64, error) {
resBytes, err := mpv.ExecuteIPC(&MPVRequest{
Command: []string{"get_property", "playback-time"},
})
if err != nil {
return 0, err
}
var res MPVResponsePlayback
if err = json.Unmarshal(resBytes, &res); err != nil {
return 0, err
}
return res.Data, nil
}
func ParseWebMedia(url string) (string, error) {
res, err := http.Get(url)
if err != nil {
return "", err
}
defer res.Body.Close()
scripts := make([]string, 0)
var processNode func(*html.Node)
processNode = func(n *html.Node) {
if n.Type == html.ElementNode && n.Data == "script" {
if n.FirstChild != nil && n.FirstChild.Type == html.TextNode {
scripts = append(scripts, n.FirstChild.Data)
}
}
for c := n.FirstChild; c != nil; c = c.NextSibling {
processNode(c)
}
}
node, err := html.Parse(res.Body)
if err != nil {
return "", err
}
processNode(node)
var streamChannels string
for _, script := range scripts {
if strings.Contains(script, "var streamChannels") {
streamChannels = script
break
}
}
if streamChannels == "" {
return "", fmt.Errorf("failed to find streamChannels")
}
re := regexp.MustCompile(`url:.*'(.*)'`)
if match := re.FindAllStringSubmatch(streamChannels, 1); match != nil {
return match[0][1], nil
} else {
return "", fmt.Errorf("regex failed")
}
}
const MPTV_SPAWN_GRACE = 5 * time.Second
const MPTV_INQUIRE_INTERVAL = time.Second
const MPTV_MAX_ATTEMPTS = 1
var web string
var socketPath string
func main() {
flag.StringVar(&web, "web", "", "web media")
flag.StringVar(&socketPath, "sock", "", "where to place socket")
flag.Parse()
streamUrl, err := ParseWebMedia(web)
if err != nil {
fmt.Fprintf(os.Stderr, "failed to get stream: %s\n", err)
return
}
mpv := NewMPV(streamUrl, socketPath)
if err = mpv.Spawn(); err != nil {
fmt.Fprintln(os.Stderr, err)
return
}
time.Sleep(MPTV_SPAWN_GRACE)
playback := 0.0
lastPlayback := 0.0
attempt := 0
for {
// inquire
playback, err = mpv.InquirePlayback()
if err != nil {
// dead mpv, restart
mpv.Stop()
streamUrl, err = ParseWebMedia(web)
if err != nil {
fmt.Fprintf(os.Stderr, "failed to get stream: %s\n", err)
// dont exit loop here since web request may fail due internet outage
}
mpv = NewMPV(streamUrl, socketPath)
if err = mpv.Spawn(); err != nil {
// failed to spawn mpv this time? fatal
fmt.Fprintln(os.Stderr, err)
return
}
// reset values
playback = 0.0
lastPlayback = 0.0
attempt = 0
// skip this cycle
time.Sleep(MPTV_SPAWN_GRACE)
continue
}
if lastPlayback == 0.0 {
// first init of last playback, dont count as attempt
lastPlayback = playback
} else if playback == lastPlayback {
// playback stuck, increment attempt
attempt += 1
if attempt > MPTV_MAX_ATTEMPTS {
// attempts exceeded, shoot in the head old mpv
mpv.Stop()
// respawn mpv
streamUrl, err = ParseWebMedia(web)
if err != nil {
fmt.Fprintf(os.Stderr, "failed to get stream: %s\n", err)
// dont exit loop here since web request may fail due internet outage
}
mpv = NewMPV(streamUrl, socketPath)
if err = mpv.Spawn(); err != nil {
// failed to spawn mpv this time? fatal
fmt.Fprintln(os.Stderr, err)
return
}
// reset values
playback = 0.0
lastPlayback = 0.0
attempt = 0
// skip this cycle
time.Sleep(MPTV_SPAWN_GRACE)
continue
}
} else {
// playback doesnt match last playback that was already initalized - we're good
lastPlayback = playback
attempt = 0
}
// new cycle
time.Sleep(MPTV_INQUIRE_INTERVAL)
}
}