2012-12-21 08:51:59 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright: Damien Elmes <anki@ichi2.net>
|
|
|
|
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
|
2012-12-22 01:17:10 +01:00
|
|
|
import re, sys, threading, time, subprocess, os, atexit
|
|
|
|
import random
|
|
|
|
from anki.hooks import addHook
|
2016-07-26 04:15:43 +02:00
|
|
|
from anki.utils import tmpdir, isWin, isMac, isLin
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
# Shared utils
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
_soundReg = "\[sound:(.*?)\]"
|
|
|
|
|
|
|
|
def playFromText(text):
|
|
|
|
for match in re.findall(_soundReg, text):
|
|
|
|
play(match)
|
|
|
|
|
|
|
|
def stripSounds(text):
|
|
|
|
return re.sub(_soundReg, "", text)
|
|
|
|
|
|
|
|
def hasSound(text):
|
|
|
|
return re.search(_soundReg, text) is not None
|
|
|
|
|
2016-07-26 04:15:43 +02:00
|
|
|
# Packaged commands
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
# return modified command array that points to bundled command, and return
|
|
|
|
# required environment
|
|
|
|
def _packagedCmd(cmd):
|
|
|
|
cmd = cmd[:]
|
|
|
|
env = os.environ.copy()
|
|
|
|
if isMac:
|
|
|
|
dir = os.path.dirname(os.path.abspath(__file__))
|
2017-01-09 03:52:52 +01:00
|
|
|
exeDir = os.path.abspath(dir + "/../../Resources/audio")
|
2016-07-26 04:15:43 +02:00
|
|
|
else:
|
|
|
|
exeDir = os.path.dirname(os.path.abspath(sys.argv[0]))
|
|
|
|
if isWin and not cmd[0].endswith(".exe"):
|
|
|
|
cmd[0] += ".exe"
|
|
|
|
path = os.path.join(exeDir, cmd[0])
|
|
|
|
if not os.path.exists(path):
|
|
|
|
return cmd, env
|
|
|
|
cmd[0] = path
|
|
|
|
# need to set lib path for linux
|
|
|
|
if isLin:
|
|
|
|
env["LD_LIBRARY_PATH"] = exeDir
|
|
|
|
return cmd, env
|
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
##########################################################################
|
|
|
|
|
2016-05-12 06:45:35 +02:00
|
|
|
processingSrc = "rec.wav"
|
|
|
|
processingDst = "rec.mp3"
|
2012-12-21 08:51:59 +01:00
|
|
|
processingChain = []
|
|
|
|
recFiles = []
|
|
|
|
|
|
|
|
processingChain = [
|
|
|
|
["lame", "rec.wav", processingDst, "--noreplaygain", "--quiet"],
|
|
|
|
]
|
|
|
|
|
|
|
|
# don't show box on windows
|
|
|
|
if isWin:
|
|
|
|
si = subprocess.STARTUPINFO()
|
|
|
|
try:
|
|
|
|
si.dwFlags |= subprocess.STARTF_USESHOWWINDOW
|
|
|
|
except:
|
|
|
|
# python2.7+
|
|
|
|
si.dwFlags |= subprocess._subprocess.STARTF_USESHOWWINDOW
|
|
|
|
else:
|
|
|
|
si = None
|
|
|
|
|
|
|
|
def retryWait(proc):
|
|
|
|
# osx throws interrupted system call errors frequently
|
|
|
|
while 1:
|
|
|
|
try:
|
|
|
|
return proc.wait()
|
|
|
|
except OSError:
|
|
|
|
continue
|
|
|
|
|
|
|
|
# Mplayer settings
|
|
|
|
##########################################################################
|
|
|
|
|
2016-07-26 04:15:43 +02:00
|
|
|
mplayerCmd = ["mplayer", "-really-quiet", "-noautosub"]
|
2012-12-21 08:51:59 +01:00
|
|
|
if isWin:
|
2016-07-26 04:15:43 +02:00
|
|
|
mplayerCmd += ["-ao", "win32"]
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
# Mplayer in slave mode
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
mplayerQueue = []
|
|
|
|
mplayerManager = None
|
|
|
|
mplayerReader = None
|
|
|
|
mplayerEvt = threading.Event()
|
|
|
|
mplayerClear = False
|
|
|
|
|
|
|
|
class MplayerMonitor(threading.Thread):
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
global mplayerClear
|
|
|
|
self.mplayer = None
|
|
|
|
self.deadPlayers = []
|
|
|
|
while 1:
|
|
|
|
mplayerEvt.wait()
|
|
|
|
mplayerEvt.clear()
|
|
|
|
# clearing queue?
|
|
|
|
if mplayerClear and self.mplayer:
|
|
|
|
try:
|
2016-12-17 04:47:07 +01:00
|
|
|
self.mplayer.stdin.write(b"stop\n")
|
2016-07-12 08:55:10 +02:00
|
|
|
self.mplayer.stdin.flush()
|
2012-12-21 08:51:59 +01:00
|
|
|
except:
|
|
|
|
# mplayer quit by user (likely video)
|
|
|
|
self.deadPlayers.append(self.mplayer)
|
|
|
|
self.mplayer = None
|
|
|
|
# loop through files to play
|
|
|
|
while mplayerQueue:
|
|
|
|
# ensure started
|
|
|
|
if not self.mplayer:
|
|
|
|
self.startProcess()
|
|
|
|
# pop a file
|
|
|
|
try:
|
|
|
|
item = mplayerQueue.pop(0)
|
|
|
|
except IndexError:
|
|
|
|
# queue was cleared by main thread
|
|
|
|
continue
|
|
|
|
if mplayerClear:
|
|
|
|
mplayerClear = False
|
2016-07-12 08:55:10 +02:00
|
|
|
extra = b""
|
2012-12-21 08:51:59 +01:00
|
|
|
else:
|
2016-07-12 08:55:10 +02:00
|
|
|
extra = b" 1"
|
|
|
|
cmd = b'loadfile "%s"%s\n' % (item.encode("utf8"), extra)
|
2012-12-21 08:51:59 +01:00
|
|
|
try:
|
|
|
|
self.mplayer.stdin.write(cmd)
|
2016-07-12 08:55:10 +02:00
|
|
|
self.mplayer.stdin.flush()
|
2012-12-21 08:51:59 +01:00
|
|
|
except:
|
|
|
|
# mplayer has quit and needs restarting
|
|
|
|
self.deadPlayers.append(self.mplayer)
|
|
|
|
self.mplayer = None
|
|
|
|
self.startProcess()
|
|
|
|
self.mplayer.stdin.write(cmd)
|
2016-07-12 08:55:10 +02:00
|
|
|
self.mplayer.stdin.flush()
|
2012-12-21 08:51:59 +01:00
|
|
|
# if we feed mplayer too fast it loses files
|
|
|
|
time.sleep(1)
|
|
|
|
# wait() on finished processes. we don't want to block on the
|
|
|
|
# wait, so we keep trying each time we're reactivated
|
|
|
|
def clean(pl):
|
|
|
|
if pl.poll() is not None:
|
|
|
|
pl.wait()
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
self.deadPlayers = [pl for pl in self.deadPlayers if clean(pl)]
|
|
|
|
|
|
|
|
def kill(self):
|
|
|
|
if not self.mplayer:
|
|
|
|
return
|
|
|
|
try:
|
2016-12-17 04:47:07 +01:00
|
|
|
self.mplayer.stdin.write(b"quit\n")
|
2016-07-12 08:55:10 +02:00
|
|
|
self.mplayer.stdin.flush()
|
2012-12-21 08:51:59 +01:00
|
|
|
self.deadPlayers.append(self.mplayer)
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
self.mplayer = None
|
|
|
|
|
|
|
|
def startProcess(self):
|
|
|
|
try:
|
|
|
|
cmd = mplayerCmd + ["-slave", "-idle"]
|
2016-07-26 04:15:43 +02:00
|
|
|
cmd, env = _packagedCmd(cmd)
|
2012-12-21 08:51:59 +01:00
|
|
|
self.mplayer = subprocess.Popen(
|
|
|
|
cmd, startupinfo=si, stdin=subprocess.PIPE,
|
2016-05-31 09:51:16 +02:00
|
|
|
stdout=subprocess.DEVNULL, stderr=subprocess.DEVNULL,
|
2016-07-12 08:55:10 +02:00
|
|
|
env=env)
|
2012-12-21 08:51:59 +01:00
|
|
|
except OSError:
|
|
|
|
mplayerEvt.clear()
|
|
|
|
raise Exception("Did you install mplayer?")
|
|
|
|
|
|
|
|
def queueMplayer(path):
|
|
|
|
ensureMplayerThreads()
|
|
|
|
if isWin and os.path.exists(path):
|
|
|
|
# mplayer on windows doesn't like the encoding, so we create a
|
|
|
|
# temporary file instead. oddly, foreign characters in the dirname
|
|
|
|
# don't seem to matter.
|
|
|
|
dir = tmpdir()
|
|
|
|
name = os.path.join(dir, "audio%s%s" % (
|
|
|
|
random.randrange(0, 1000000), os.path.splitext(path)[1]))
|
|
|
|
f = open(name, "wb")
|
|
|
|
f.write(open(path, "rb").read())
|
|
|
|
f.close()
|
|
|
|
# it wants unix paths, too!
|
|
|
|
path = name.replace("\\", "/")
|
|
|
|
mplayerQueue.append(path)
|
|
|
|
mplayerEvt.set()
|
|
|
|
|
|
|
|
def clearMplayerQueue():
|
|
|
|
global mplayerClear, mplayerQueue
|
|
|
|
mplayerQueue = []
|
|
|
|
mplayerClear = True
|
|
|
|
mplayerEvt.set()
|
|
|
|
|
|
|
|
def ensureMplayerThreads():
|
|
|
|
global mplayerManager
|
|
|
|
if not mplayerManager:
|
|
|
|
mplayerManager = MplayerMonitor()
|
|
|
|
mplayerManager.daemon = True
|
|
|
|
mplayerManager.start()
|
|
|
|
# ensure the tmpdir() exit handler is registered first so it runs
|
|
|
|
# after the mplayer exit
|
|
|
|
tmpdir()
|
|
|
|
# clean up mplayer on exit
|
|
|
|
atexit.register(stopMplayer)
|
|
|
|
|
|
|
|
def stopMplayer(*args):
|
|
|
|
if not mplayerManager:
|
|
|
|
return
|
|
|
|
mplayerManager.kill()
|
|
|
|
|
|
|
|
addHook("unloadProfile", stopMplayer)
|
|
|
|
|
|
|
|
# PyAudio recording
|
|
|
|
##########################################################################
|
|
|
|
|
2016-06-23 04:04:48 +02:00
|
|
|
import pyaudio
|
|
|
|
import wave
|
2014-04-17 21:17:05 +02:00
|
|
|
|
2016-06-23 04:04:48 +02:00
|
|
|
PYAU_FORMAT = pyaudio.paInt16
|
|
|
|
PYAU_CHANNELS = 1
|
|
|
|
PYAU_INPUT_INDEX = None
|
2012-12-21 08:51:59 +01:00
|
|
|
|
2017-02-06 23:21:33 +01:00
|
|
|
class _Recorder:
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
def postprocess(self, encode=True):
|
|
|
|
self.encode = encode
|
|
|
|
for c in processingChain:
|
|
|
|
#print c
|
|
|
|
if not self.encode and c[0] == 'lame':
|
|
|
|
continue
|
|
|
|
try:
|
2016-07-26 04:15:43 +02:00
|
|
|
cmd, env = _packagedCmd(c)
|
|
|
|
ret = retryWait(subprocess.Popen(cmd, startupinfo=si, env=env))
|
2012-12-21 08:51:59 +01:00
|
|
|
except:
|
|
|
|
ret = True
|
|
|
|
if ret:
|
|
|
|
raise Exception(_(
|
|
|
|
"Error running %s") %
|
2016-07-26 04:15:43 +02:00
|
|
|
" ".join(cmd))
|
2012-12-21 08:51:59 +01:00
|
|
|
|
|
|
|
class PyAudioThreadedRecorder(threading.Thread):
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
threading.Thread.__init__(self)
|
|
|
|
self.finish = False
|
|
|
|
|
|
|
|
def run(self):
|
|
|
|
chunk = 1024
|
2016-06-23 04:04:48 +02:00
|
|
|
p = pyaudio.PyAudio()
|
2014-04-17 21:17:05 +02:00
|
|
|
|
|
|
|
rate = int(p.get_default_input_device_info()['defaultSampleRate'])
|
|
|
|
|
2012-12-21 08:51:59 +01:00
|
|
|
stream = p.open(format=PYAU_FORMAT,
|
|
|
|
channels=PYAU_CHANNELS,
|
2014-04-17 21:17:05 +02:00
|
|
|
rate=rate,
|
2012-12-21 08:51:59 +01:00
|
|
|
input=True,
|
|
|
|
input_device_index=PYAU_INPUT_INDEX,
|
|
|
|
frames_per_buffer=chunk)
|
2014-04-17 21:17:05 +02:00
|
|
|
|
2016-05-31 09:51:16 +02:00
|
|
|
data = b""
|
2012-12-21 08:51:59 +01:00
|
|
|
while not self.finish:
|
|
|
|
try:
|
2016-05-31 09:51:16 +02:00
|
|
|
data += stream.read(chunk)
|
2016-05-12 06:45:35 +02:00
|
|
|
except IOError as e:
|
2012-12-21 08:51:59 +01:00
|
|
|
if e[1] == pyaudio.paInputOverflowed:
|
2016-05-31 09:51:16 +02:00
|
|
|
pass
|
2012-12-21 08:51:59 +01:00
|
|
|
else:
|
|
|
|
raise
|
|
|
|
stream.close()
|
|
|
|
p.terminate()
|
|
|
|
wf = wave.open(processingSrc, 'wb')
|
|
|
|
wf.setnchannels(PYAU_CHANNELS)
|
|
|
|
wf.setsampwidth(p.get_sample_size(PYAU_FORMAT))
|
2014-04-17 21:17:05 +02:00
|
|
|
wf.setframerate(rate)
|
2012-12-21 08:51:59 +01:00
|
|
|
wf.writeframes(data)
|
|
|
|
wf.close()
|
|
|
|
|
|
|
|
class PyAudioRecorder(_Recorder):
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
for t in recFiles + [processingSrc, processingDst]:
|
|
|
|
try:
|
|
|
|
os.unlink(t)
|
|
|
|
except OSError:
|
|
|
|
pass
|
|
|
|
self.encode = False
|
|
|
|
|
|
|
|
def start(self):
|
|
|
|
self.thread = PyAudioThreadedRecorder()
|
|
|
|
self.thread.start()
|
|
|
|
|
|
|
|
def stop(self):
|
|
|
|
self.thread.finish = True
|
|
|
|
self.thread.join()
|
|
|
|
|
|
|
|
def file(self):
|
|
|
|
if self.encode:
|
2016-05-12 06:45:35 +02:00
|
|
|
tgt = "rec%d.mp3" % time.time()
|
2012-12-21 08:51:59 +01:00
|
|
|
os.rename(processingDst, tgt)
|
|
|
|
return tgt
|
|
|
|
else:
|
|
|
|
return processingSrc
|
|
|
|
|
|
|
|
# Audio interface
|
|
|
|
##########################################################################
|
|
|
|
|
|
|
|
_player = queueMplayer
|
|
|
|
_queueEraser = clearMplayerQueue
|
|
|
|
|
|
|
|
def play(path):
|
|
|
|
_player(path)
|
|
|
|
|
|
|
|
def clearAudioQueue():
|
|
|
|
_queueEraser()
|
|
|
|
|
|
|
|
Recorder = PyAudioRecorder
|