mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
142 lines
3.2 KiB
Go
142 lines
3.2 KiB
Go
package mpv
|
|
|
|
import (
|
|
"context"
|
|
"crypto/rand"
|
|
"encoding/hex"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/navidrome/navidrome/conf"
|
|
"github.com/navidrome/navidrome/log"
|
|
)
|
|
|
|
// mpv --no-audio-display --pause 'Jack Johnson/On And On/01 Times Like These.m4a' --input-ipc-server=/tmp/gonzo.socket
|
|
const (
|
|
mpvComdTemplate = "mpv --audio-device=%d --no-audio-display --pause %f --input-ipc-server=%s"
|
|
)
|
|
|
|
func start(args []string) (Executor, error) {
|
|
log.Debug("Executing mpv command", "cmd", args)
|
|
j := Executor{args: args}
|
|
j.PipeReader, j.out = io.Pipe()
|
|
err := j.start()
|
|
if err != nil {
|
|
return Executor{}, err
|
|
}
|
|
go j.wait()
|
|
return j, nil
|
|
}
|
|
|
|
func (j *Executor) Cancel() error {
|
|
if j.cmd != nil {
|
|
return j.cmd.Cancel()
|
|
}
|
|
return fmt.Errorf("there is non command to cancel")
|
|
}
|
|
|
|
type Executor struct {
|
|
*io.PipeReader
|
|
out *io.PipeWriter
|
|
args []string
|
|
cmd *exec.Cmd
|
|
ctx context.Context
|
|
}
|
|
|
|
func (j *Executor) start() error {
|
|
ctx := context.Background()
|
|
j.ctx = ctx
|
|
cmd := exec.CommandContext(ctx, j.args[0], j.args[1:]...) // #nosec
|
|
cmd.Stdout = j.out
|
|
if log.IsGreaterOrEqualTo(log.LevelTrace) {
|
|
cmd.Stderr = os.Stderr
|
|
} else {
|
|
cmd.Stderr = io.Discard
|
|
}
|
|
j.cmd = cmd
|
|
|
|
if err := cmd.Start(); err != nil {
|
|
return fmt.Errorf("starting cmd: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (j *Executor) wait() {
|
|
if err := j.cmd.Wait(); err != nil {
|
|
var exitErr *exec.ExitError
|
|
if errors.As(err, &exitErr) {
|
|
_ = j.out.CloseWithError(fmt.Errorf("%s exited with non-zero status code: %d", j.args[0], exitErr.ExitCode()))
|
|
} else {
|
|
_ = j.out.CloseWithError(fmt.Errorf("waiting %s cmd: %w", j.args[0], err))
|
|
}
|
|
return
|
|
}
|
|
_ = j.out.Close()
|
|
}
|
|
|
|
// Path will always be an absolute path
|
|
func createMPVCommand(cmd, deviceName string, filename string, socketName string) []string {
|
|
split := strings.Split(fixCmd(cmd), " ")
|
|
for i, s := range split {
|
|
s = strings.ReplaceAll(s, "%d", deviceName)
|
|
s = strings.ReplaceAll(s, "%f", filename)
|
|
s = strings.ReplaceAll(s, "%s", socketName)
|
|
split[i] = s
|
|
}
|
|
|
|
return split
|
|
}
|
|
|
|
func fixCmd(cmd string) string {
|
|
split := strings.Split(cmd, " ")
|
|
var result []string
|
|
cmdPath, _ := mpvCommand()
|
|
for _, s := range split {
|
|
if s == "mpv" || s == "mpv.exe" {
|
|
result = append(result, cmdPath)
|
|
} else {
|
|
result = append(result, s)
|
|
}
|
|
}
|
|
return strings.Join(result, " ")
|
|
}
|
|
|
|
// This is a 1:1 copy of the stuff in ffmpeg.go, need to be unified.
|
|
func mpvCommand() (string, error) {
|
|
mpvOnce.Do(func() {
|
|
if conf.Server.MPVPath != "" {
|
|
mpvPath = conf.Server.MPVPath
|
|
mpvPath, mpvErr = exec.LookPath(mpvPath)
|
|
} else {
|
|
mpvPath, mpvErr = exec.LookPath("mpv")
|
|
if errors.Is(mpvErr, exec.ErrDot) {
|
|
log.Trace("mpv found in current folder '.'")
|
|
mpvPath, mpvErr = exec.LookPath("./mpv")
|
|
}
|
|
}
|
|
if mpvErr == nil {
|
|
log.Info("Found mpv", "path", mpvPath)
|
|
return
|
|
}
|
|
})
|
|
return mpvPath, mpvErr
|
|
}
|
|
|
|
var (
|
|
mpvOnce sync.Once
|
|
mpvPath string
|
|
mpvErr error
|
|
)
|
|
|
|
func TempFileName(prefix, suffix string) string {
|
|
randBytes := make([]byte, 16)
|
|
// we can savely ignore the return value since we're loading into a precreated, fixedsized buffer
|
|
_, _ = rand.Read(randBytes)
|
|
return filepath.Join(os.TempDir(), prefix+hex.EncodeToString(randBytes)+suffix)
|
|
}
|