package main import ( "fmt" "io" "log" "net" "os" "os/signal" "strings" "time" "github.com/fsnotify/fsnotify" "github.com/godbus/dbus/v5" mpris2 "github.com/hrfee/mpris2client" flag "github.com/spf13/pflag" ) // Various paths and values to use elsewhere. const ( SOCK = "/tmp/waybar-mpris.sock" LOGFILE = "/tmp/waybar-mpris.log" OUTFILE = "/tmp/waybar-mpris.out" POLL = 1 ) // Mostly default values for flag options. var ( PLAY = "▶" PAUSE = "" SEP = " - " ORDER = "SYMBOL:ARTIST:ALBUM:TITLE:POSITION" AUTOFOCUS = false // Available commands that can be sent to running instances. COMMANDS = []string{"player-next", "player-prev", "next", "prev", "toggle", "list"} SHOW_POS = false INTERPOLATE = false REPLACE = false isSharing = false WRITER io.Writer = os.Stdout ) const ( cPlayerNext = "pn" cPlayerPrev = "pp" cNext = "mn" cPrev = "mp" cToggle = "mt" cList = "ls" cShare = "sh" cPreShare = "ps" rSuccess = "sc" rInvalidCommand = "iv" ) func stringToCmd(str string) string { switch str { case "player-next": return cPlayerNext case "player-prev": return cPlayerPrev case "next": return cNext case "prev": return cPrev case "toggle": return cToggle case "list": return cList case "share": return cShare case "pre-share": return cPreShare } return "" } // JSON returns json for waybar to consume. func playerJSON(p *mpris2.Player) string { symbol := PLAY out := "{\"class\": \"" if p.Playing { symbol = PAUSE out += "playing" } else { out += "paused" } var pos string if SHOW_POS { pos = p.StringPosition() if pos != "" { pos = "(" + pos + ")" } } var items []string order := strings.Split(ORDER, ":") for _, v := range order { switch v { case "SYMBOL": items = append(items, symbol) case "ARTIST": if p.Artist != "" { items = append(items, p.Artist) } case "ALBUM": if p.Album != "" { items = append(items, p.Album) } case "TITLE": if p.Title != "" { items = append(items, p.Title) } case "POSITION": if pos != "" && SHOW_POS { items = append(items, pos) } case "PLAYER": if p.Name != "" { items = append(items, p.Name) } } } if len(items) == 0 { return "{}" } text := "" for i, v := range items { right := "" if (v == symbol || v == pos) && i != len(items)-1 { right = " " } else if i != len(items)-1 && items[i+1] != symbol && items[i+1] != pos { right = SEP } else { right = " " } text += v + right } out += "\",\"text\":\"" + text + "\"," out += "\"tooltip\":\"" + strings.ReplaceAll(p.Title, "&", "&") + "\\n" if p.Artist != "" { out += "by " + strings.ReplaceAll(p.Artist, "&", "&") + "\\n" } if p.Album != "" { out += "from " + strings.ReplaceAll(p.Album, "&", "&") + "\\n" } out += "(" + p.Name + ")\"}" return out // return fmt.Sprintf("{\"class\":\"%s\",\"text\":\"%s\",\"tooltip\":\"%s\"}", data["class"], data["text"], data["tooltip"]) // out, err := json.Marshal(data) // if err != nil { // return "{}" // } // return string(out) } type players struct { mpris2 *mpris2.Mpris2 } func (pl *players) JSON() string { if len(pl.mpris2.List) != 0 { return playerJSON(pl.mpris2.List[pl.mpris2.Current]) } return "{}" } func (pl *players) Next() { pl.mpris2.List[pl.mpris2.Current].Next() } func (pl *players) Prev() { pl.mpris2.List[pl.mpris2.Current].Previous() } func (pl *players) Toggle() { pl.mpris2.List[pl.mpris2.Current].Toggle() } func execCommand(cmd string) { conn, err := net.Dial("unix", SOCK) if err != nil { log.Fatalln("Couldn't dial:", err) } shortCmd := stringToCmd(cmd) _, err = conn.Write([]byte(shortCmd)) if err != nil { log.Fatalln("Couldn't send command") } fmt.Println("Sent.") if cmd == "list" { buf := make([]byte, 512) nr, err := conn.Read(buf) if err != nil { log.Fatalln("Couldn't read response.") } response := string(buf[0:nr]) fmt.Println("Response:") fmt.Printf(response) } os.Exit(0) } func duplicateOutput() error { // Print to stderr to avoid errors from waybar os.Stderr.WriteString("waybar-mpris is already running. This instance will clone its output.") conn, err := net.Dial("unix", SOCK) if err != nil { return err } _, err = conn.Write([]byte(cPreShare)) if err != nil { log.Fatalf("Couldn't send command: %v", err) return err } buf := make([]byte, 512) nr, err := conn.Read(buf) if err != nil { log.Fatalf("Couldn't read response: %v", err) return err } argString := "" for _, arg := range os.Args { argString += arg + "|" } if string(buf[0:nr]) == argString { conn.Close() conn, err = net.Dial("unix", SOCK) if err != nil { return err } // Tell other instance to share output in OUTFILE _, err := conn.Write([]byte(cShare)) if err != nil { log.Fatalf("Couldn't send command: %v", err) } buf := make([]byte, 2) nr, err := conn.Read(buf) if err != nil { log.Fatalf("Couldn't read response: %v", err) } if resp := string(buf[0:nr]); resp == rSuccess { // t, err := tail.TailFile(OUTFILE, tail.Config{ // Follow: true, // MustExist: true, // Logger: tail.DiscardingLogger, // }) // if err == nil { // for line := range t.Lines { // fmt.Println(line.Text) // } // } f, err := os.Open(OUTFILE) if err != nil { log.Fatalf("Failed to open \"%s\": %v", OUTFILE, err) } watcher, err := fsnotify.NewWatcher() if err != nil { log.Fatalf("Failed to start watcher: %v", err) } defer watcher.Close() err = watcher.Add(OUTFILE) if err != nil { log.Fatalf("Failed to watch file: %v", err) } for { select { case event, ok := <-watcher.Events: if !ok { log.Printf("Watcher failed: %v", err) return err } if event.Op&fsnotify.Write == fsnotify.Write { l, err := io.ReadAll(f) if err != nil { log.Printf("Failed to read file: %v", err) return err } str := string(l) // Trim extra newline is necessary if str[len(str)-2:] == "\n\n" { fmt.Print(str[:len(str)-1]) } else { fmt.Print(str) } f.Seek(0, 0) } } } } } return nil } func listenForCommands(players *players) { listener, err := net.Listen("unix", SOCK) c := make(chan os.Signal, 1) signal.Notify(c, os.Interrupt) go func() { <-c os.Remove(OUTFILE) os.Remove(SOCK) os.Exit(1) }() if err != nil { log.Fatalf("Couldn't establish socket connection at %s (error %s)\n", SOCK, err) } defer func() { listener.Close() os.Remove(SOCK) }() for { con, err := listener.Accept() if err != nil { log.Println("Couldn't accept:", err) continue } buf := make([]byte, 2) nr, err := con.Read(buf) if err != nil { log.Println("Couldn't read:", err) continue } command := string(buf[0:nr]) switch command { case cPlayerNext: length := len(players.mpris2.List) if length != 1 { if players.mpris2.Current < uint(length-1) { players.mpris2.Current++ } else { players.mpris2.Current = 0 } players.mpris2.Refresh() } case cPlayerPrev: length := len(players.mpris2.List) if length != 1 { if players.mpris2.Current != 0 { players.mpris2.Current-- } else { players.mpris2.Current = uint(length - 1) } players.mpris2.Refresh() } case cNext: players.Next() case cPrev: players.Prev() case cToggle: players.Toggle() case cList: con.Write([]byte(players.mpris2.String())) case cShare: if !isSharing { f, err := os.OpenFile(OUTFILE, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0666) defer f.Close() if err != nil { fmt.Fprintf(con, "Failed: %v", err) } var out io.Writer = emptyEveryWrite{file: f} WRITER = io.MultiWriter(os.Stdout, out) isSharing = true } fmt.Fprint(con, rSuccess) /* Prior to sharing, the first instance sends its os.Args. If the second instances args are different, the first sends the raw data (artist, album, etc.) If they are the same, the first instance just sends its output and the second prints it. */ case cPreShare: out := "" for _, arg := range os.Args { out += arg + "|" } con.Write([]byte(out)) default: fmt.Println("Invalid command") } con.Close() } } type emptyEveryWrite struct { file *os.File } func (w emptyEveryWrite) Write(p []byte) (n int, err error) { offset, err := w.file.Seek(0, 0) if err != nil { return 0, err } return w.file.WriteAt(p, offset) } func main() { logfile, err := os.OpenFile(LOGFILE, os.O_CREATE|os.O_APPEND|os.O_RDWR, 0666) if err != nil { log.Fatalf("Couldn't open %s for writing: %s", LOGFILE, err) } mw := io.MultiWriter(logfile, os.Stdout) log.SetOutput(mw) flag.StringVar(&PLAY, "play", PLAY, "Play symbol/text to use.") flag.StringVar(&PAUSE, "pause", PAUSE, "Pause symbol/text to use.") flag.StringVar(&SEP, "separator", SEP, "Separator string to use between artist, album, and title.") flag.StringVar(&ORDER, "order", ORDER, "Element order. An extra \"PLAYER\" element is also available.") flag.BoolVar(&AUTOFOCUS, "autofocus", AUTOFOCUS, "Auto switch to currently playing music players.") flag.BoolVar(&SHOW_POS, "position", SHOW_POS, "Show current position between brackets, e.g (04:50/05:00)") flag.BoolVar(&INTERPOLATE, "interpolate", INTERPOLATE, "Interpolate track position (helpful for players that don't update regularly, e.g mpDris2)") flag.BoolVar(&REPLACE, "replace", REPLACE, "replace existing waybar-mpris if found. When false, new instance will clone the original instances output.") var command string flag.StringVar(&command, "send", "", "send command to already runnning waybar-mpris instance. (options: "+strings.Join(COMMANDS, "/")+")") flag.Parse() os.Stderr = logfile if command != "" { execCommand(command) } // fmt.Println("New array", players) // Start command listener if _, err := os.Stat(SOCK); err == nil { if REPLACE { fmt.Printf("Socket %s already exists, this could mean waybar-mpris is already running.\nStarting this instance will overwrite the file, possibly stopping other instances from accepting commands.\n", SOCK) var input string ignoreChoice := false fmt.Printf("Continue? [y/n]: ") go func() { fmt.Scanln(&input) if strings.Contains(input, "y") && !ignoreChoice { os.Remove(SOCK) } }() time.Sleep(5 * time.Second) if input == "" { fmt.Printf("\nRemoving due to lack of input.\n") ignoreChoice = true // os.Remove(SOCK) } // When waybar-mpris is already running, we attach to its output instead of launching a whole new instance. } else if err := duplicateOutput(); err != nil { os.Stdout.WriteString("Couldn't dial socket, deleting instead: " + err.Error()) os.Remove(SOCK) os.Remove(OUTFILE) } } conn, err := dbus.SessionBus() if err != nil { log.Fatalln("Error connecting to DBus:", err) } players := &players{ mpris2: mpris2.NewMpris2(conn, INTERPOLATE, POLL, AUTOFOCUS), } players.mpris2.Reload() players.mpris2.Sort() lastLine := "" go listenForCommands(players) go players.mpris2.Listen() if SHOW_POS { go func() { for { time.Sleep(POLL * time.Second) if len(players.mpris2.List) != 0 { if players.mpris2.List[players.mpris2.Current].Playing { go fmt.Fprintln(WRITER, players.JSON()) } } } }() } fmt.Fprintln(WRITER, players.JSON()) for v := range players.mpris2.Messages { if v.Name == "refresh" { if AUTOFOCUS { players.mpris2.Sort() } if l := players.JSON(); l != lastLine { lastLine = l fmt.Fprintln(WRITER, l) } } } players.mpris2.Refresh() }