2013-07-28 20:44:16 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
|
|
|
"""
|
|
|
|
This script emulates "unique application" functionality on Linux. When starting
|
|
|
|
playback with this script, it will try to reuse an already running instance of
|
2014-02-05 18:01:06 +00:00
|
|
|
mpv (but only if that was started with umpv). Other mpv instances (not started
|
|
|
|
by umpv) are ignored, and the script doesn't know about them.
|
2013-07-28 20:44:16 +00:00
|
|
|
|
2014-02-05 18:01:06 +00:00
|
|
|
This only takes filenames as arguments. Custom options can't be used; the script
|
|
|
|
interprets them as filenames. If mpv is already running, the files passed to
|
|
|
|
umpv are appended to mpv's internal playlist. If a file does not exist or is
|
|
|
|
otherwise not playable, mpv will skip the playlist entry when attempting to
|
|
|
|
play it (from the GUI perspective, it's silently ignored).
|
2013-07-28 20:44:16 +00:00
|
|
|
|
|
|
|
If mpv isn't running yet, this script will start mpv and let it control the
|
2014-02-05 18:04:35 +00:00
|
|
|
current terminal. It will not write output to stdout/stderr, because this
|
|
|
|
will typically just fill ~/.xsession-errors with garbage.
|
2013-07-28 20:44:16 +00:00
|
|
|
|
|
|
|
mpv will terminate if there are no more files to play, and running the umpv
|
|
|
|
script after that will start a new mpv instance.
|
|
|
|
|
|
|
|
Note that you can control the mpv instance by writing to the command fifo:
|
|
|
|
|
|
|
|
echo "cycle fullscreen" > /tmp/umpv-fifo-*
|
|
|
|
|
2014-04-28 23:23:44 +00:00
|
|
|
Note: you can supply custom options with the UMPV_OPTIONS environment variable.
|
|
|
|
The environment variable will be split on whitespace, and each item is
|
|
|
|
passed as option to mpv _if_ the script starts mpv. If mpv is not started
|
|
|
|
by the script (i.e. mpv is already running), the options will be ignored.
|
|
|
|
|
2013-07-28 20:44:16 +00:00
|
|
|
Warning:
|
|
|
|
|
|
|
|
The script attempts to make sure the FIFO is safely created (i.e. not world-
|
2014-02-05 18:01:06 +00:00
|
|
|
writable), and checks that it's really a FIFO. This is important for security,
|
|
|
|
because the FIFO allows anyone with write access to run arbitrary commands
|
|
|
|
in mpv's context using the "run" input command. If you are worried about
|
|
|
|
security, you should verify that the code handles these concerns correctly.
|
2013-07-28 20:44:16 +00:00
|
|
|
"""
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import os
|
|
|
|
import errno
|
|
|
|
import subprocess
|
|
|
|
import fcntl
|
|
|
|
import stat
|
2014-04-29 00:16:18 +00:00
|
|
|
import string
|
2013-07-28 20:44:16 +00:00
|
|
|
|
2014-04-29 00:06:40 +00:00
|
|
|
if len(sys.argv) < 1:
|
2013-07-28 20:44:16 +00:00
|
|
|
sys.exit(1)
|
|
|
|
files = sys.argv[1:]
|
2014-04-29 00:16:18 +00:00
|
|
|
|
|
|
|
# this is about the same method mpv uses to decide this
|
|
|
|
def is_url(filename):
|
|
|
|
parts = filename.split(":", 1)
|
|
|
|
if len(parts) < 2:
|
|
|
|
return False
|
|
|
|
# protocol prefix has no special characters => it's an URL
|
|
|
|
prefix = parts[0]
|
|
|
|
return len(prefix.translate(None, string.letters)) == 0
|
|
|
|
|
2013-07-28 20:44:16 +00:00
|
|
|
# make them absolute; also makes them safe against interpretation as options
|
2014-04-29 00:16:18 +00:00
|
|
|
def make_abs(filename):
|
|
|
|
if not is_url(filename):
|
|
|
|
return os.path.abspath(f)
|
|
|
|
return f
|
|
|
|
files = [make_abs(f) for f in files]
|
2013-07-28 20:44:16 +00:00
|
|
|
|
2014-04-28 23:23:44 +00:00
|
|
|
opts_env = os.getenv("UMPV_OPTIONS")
|
|
|
|
opts_env = opts_env.split() if opts_env else []
|
|
|
|
|
2013-07-28 20:44:16 +00:00
|
|
|
FIFO = "/tmp/umpv-fifo-" + os.getenv("USER")
|
|
|
|
|
|
|
|
fifo_fd = -1
|
|
|
|
try:
|
|
|
|
fifo_fd = os.open(FIFO, os.O_NONBLOCK | os.O_WRONLY)
|
|
|
|
except OSError as e:
|
|
|
|
if e.errno == errno.ENXIO:
|
|
|
|
pass # pipe has no writer
|
|
|
|
elif e.errno == errno.ENOENT:
|
|
|
|
pass # doesn't exist
|
|
|
|
else:
|
|
|
|
raise e
|
|
|
|
|
|
|
|
if fifo_fd >= 0:
|
|
|
|
st = os.fstat(fifo_fd)
|
|
|
|
if (((st.st_mode & (stat.S_IRWXG | stat.S_IRWXO)) != 0) or
|
|
|
|
(not stat.S_ISFIFO(st.st_mode)) or
|
|
|
|
(st.st_uid != os.getuid())):
|
|
|
|
print("error: command FIFO is not a FIFO or has bogus permissions")
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
if fifo_fd >= 0:
|
|
|
|
# Unhandled race condition: what if mpv is terminating right now?
|
|
|
|
fcntl.fcntl(fifo_fd, fcntl.F_SETFL, 0) # set blocking mode
|
|
|
|
fifo = os.fdopen(fifo_fd, "w")
|
|
|
|
for f in files:
|
|
|
|
# escape: \ \n "
|
|
|
|
f = f.replace("\\", "\\\\").replace("\"", "\\\"").replace("\n", "\\n")
|
|
|
|
f = "\"" + f + "\""
|
|
|
|
fifo.write("raw loadfile " + f + " append\n")
|
|
|
|
else:
|
|
|
|
# Recreate pipe if it doesn't already exist.
|
|
|
|
# Also makes sure it's safe, and no other user can create a bogus pipe
|
|
|
|
# that breaks security.
|
|
|
|
try:
|
|
|
|
os.unlink(FIFO)
|
|
|
|
except OSError as e:
|
|
|
|
if e.errno == errno.ENOENT:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
raise e
|
2014-04-28 23:21:21 +00:00
|
|
|
os.mkfifo(FIFO, int("0600", 8))
|
2013-07-28 20:44:16 +00:00
|
|
|
|
2014-04-29 00:09:59 +00:00
|
|
|
opts = ["mpv", "--no-terminal", "--force-window", "--input-file=" + FIFO]
|
2014-04-28 23:23:44 +00:00
|
|
|
opts.extend(opts_env)
|
|
|
|
opts.append("--")
|
|
|
|
opts.extend(files)
|
|
|
|
|
|
|
|
subprocess.check_call(opts)
|