3 __author__ = "Artur Duque de Souza"
4 __author_email__ = "artur.souza@indt.org.br"
17 import lib.utils as utils
18 import lib.server as server
19 import plugins.transcoders.mencoder_lib.mythtv as mythtv
21 from select import select
22 import lib.transcoder as transcoder
24 __all__ = ("TranscoderMencoder",)
26 class TranscoderMencoder(transcoder.Transcoder):
27 """Transcoder class that implements a transcoder using Mencoder"""
28 mencoder_path = utils.which("mencoder")
35 # only works with avi container
38 def _setup_params(self):
39 params_first = self.params_first
42 self.args["local"] = params_first("local", False)
43 self.args["language"] = params_first("language", False)
44 self.args["subtitle"] = params_first("subtitle", False)
45 self.args["outfile"] = params_first("outfile", "-")
48 self.args["type"] = params_first("type", "file")
49 self.args["input"] = params_first("uri", "-")
52 self.args["acodec"] = params_first("acodec", "mp3lame")
53 self.args["abitrate"] = params_first("abitrate", 128)
54 self.args["volume"] = params_first("volume", 9)
57 self.args["fps"] = params_first("fps", 25)
58 self.args["vcodec"] = params_first("vcodec", "msmpeg4v2")
59 self.args["vbitrate"] = params_first("vbitrate", 500)
60 self.args["width"] = params_first("width", 400)
61 self.args["height"] = params_first("height", 240)
65 def _setup_audio(self):
66 if self.args["acodec"] == "mp3lame":
67 audio = "-oac mp3lame -lameopts cbr:br=%s vol=%s" % (
68 self.args["abitrate"], self.args["volume"])
70 audio = "-oac lavc -lavcopts acodec=%s:abitrate=%s" % (
71 self.args["acodec"], self.args["abitrate"])
77 def _setup_video(self):
78 #video = " -of %s" % self.args["mux"]
79 video = " -ofps %s" % self.args["fps"]
81 vcodec = self.args["vcodec"]
82 if vcodec == "nuv" or vcodec == "xvid"\
83 or vcodec == "qtvideo" or vcodec == "copy":
84 video += " -ovc %s" % vcodec
86 video += " -ovc lavc -lavcopts vcodec=%s:vhq:vbitrate=%s" % (
87 vcodec, self.args["vbitrate"])
89 video += " -vf scale=%s:%s" % (self.args["width"], self.args["height"])
94 def _arg_append(self, args, options):
95 for arg in shlex.split(options):
99 def _setup_mencoder_opts(self, args):
100 args.append(self.mencoder_path)
102 if self.args["type"] and self.args["type"] == "tv":
103 self._arg_append(args, self.args["tv"])
104 elif self.args["outfile"] == "-" and self.args["type"]:
105 args.append(self.args["input"])
109 if self.args["language"]:
110 self._arg_append(args, "-alang %s" % self.args["language"])
112 if self.args["subtitle"]:
113 self._arg_append(args, "-slang %s" % self.args["subtitle"])
114 self._arg_append(args, "-subfps %s" % self.args["fps"])
116 self._arg_append(args, "-idx")
117 self._arg_append(args, "-cache 1024")
118 self._arg_append(args, self._setup_audio())
119 self._arg_append(args, self._setup_video())
121 self._arg_append(args, "-really-quiet")
123 if self.args["outfile"] != "-":
124 self.args["outfile"] = ".transcoded/%s" % (
125 os.path.basename(self.args["outfile"]))
127 self._arg_append(args, "-o %s" % self.args["outfile"])
128 self._arg_append(args, "2>%s" % os.devnull)
131 def _setup_filename(self):
132 """This function setups the file to encode parsing the uri.
138 If the last one is detected we have to parse the uri to find args.
139 Then we store all the args inside a dictionary: self.args['gmyth-cat']
141 _type = self.args["type"]
144 if not os.path.exists(self.args["input"]):
146 "File requested does not exist: %s." % self.args["input"]
148 self.args["input"] = "file://%s" % self.args["input"]
151 self.args["input"] = "dvd://%s" % self.args["input"]
153 elif _type == "myth":
154 self.args["gmyth-cat"] = mythtv._setup_mythfilename(self)
157 driver = self.params_first("driver", "v4l2")
158 norm = self.params_first("norm", "pal-m")
159 channel = self.params_first("channel", "13")
160 chanlist = self.params_first("chanlist", "us-bcast")
161 outfmt = self.params_first("outfmt", "yuy2")
162 vdev = self.params_first("vdev", "/dev/video0")
163 adev = self.params_first("adev", "/dev/dsp")
164 self.args["tv"] = "tv:// -v -tv driver=%s:norm=%s:channel=%s:" \
165 "chanlist=%s:width=%s:height=%s:outfmt=%s:" \
166 "device=%s:adevice=%s" % (driver, norm,
174 def __init__(self, params):
175 transcoder.Transcoder.__init__(self, params)
176 self.mencoder_opts = []
180 self._setup_filename()
181 self._setup_mencoder_opts(self.mencoder_opts)
184 self.log.error(self.tid, "Error: %s" % e)
190 def _check_opened_file(self, stdw, _stdin):
194 return open(self.args["outfile"])
196 os.write(stdw, _stdin.read(1024))
200 def _start_outfile(self, outfd):
205 filename = self.args["input"].split("://")[1]
206 _stdin = open(filename)
207 size = int(os.path.getsize(filename))
209 self.log.error(self.tid, "Error: Mencoder stdin"\
210 " setup error: %s" % e)
211 outfd.write("Error: Mencoder stdin setup error: %s" %e)
218 stdr, stdw = os.pipe()
220 if not self._run_mencoder(input=stdr):
223 stdout = self._check_opened_file(stdw, _stdin)
227 while self.proc and self.proc.poll() == None:
229 data_in = _stdin.read(4096)
231 os.write(stdw, data_in)
233 d = stdout.read(4096)
234 self.status = utils.progress_bar(total_read,
241 d = stdout.read(4096)
244 self.log.error(self.tid, "Error: %s" % e)
248 self.log.info(self.tid, "OK: Done")
253 def _start(self, outfd):
254 # Play a file on disk or DVD
255 if not self._run_mencoder(output=subprocess.PIPE):
259 while self.proc and self.proc.poll() == None:
260 d = self.proc.stdout.read(1024)
263 self.log.error(self.tid, "Error: %s" % e)
266 self.log.info(self.tid, "OK: Done")
270 def _run_mencoder(self, input=None, output=None):
272 print self.mencoder_opts
273 self.proc = subprocess.Popen(self.mencoder_opts, stdin=input,
274 stdout=output, close_fds=True)
276 self.log.error(self.tid, "Error: Mencoder: %s" % e)
282 def start(self, outfd):
283 cmd = " ".join(self.mencoder_opts)
284 self.log.debug(self.tid, "Plugin's tid: %s" % self.tid)
285 self.log.debug(self.tid, "Mencoder: %s" % cmd)
289 if self.args["outfile"] == "-" and \
290 self.args["type"] in ["file", "dvd", "tv"]:
291 ret = self._start(outfd)
293 elif self.args["type"] == "myth":
294 ret = mythtv.start_myth(self, outfd)
297 ret = self._start_outfile(outfd)
302 self.log.error(self.tid, "Error: Problems while "\
303 "starting streaming.")
308 def _aux_stop(self, obj, next=False):
311 os.kill(obj.pid, signal.SIGKILL)
313 os.kill(obj.pid+1, signal.SIGKILL)
326 self._aux_stop(self.proc, True)
327 self._aux_stop(self.gmyth)