Use soxi instead of gstreamer to determine a track's length

Contributes-To: https://github.com/JoeLametta/whipper/issues/29
This commit is contained in:
chrysn
2016-11-07 14:57:17 +01:00
parent 5a59f3c7e3
commit ba10b573a2
2 changed files with 50 additions and 52 deletions

View File

@@ -29,7 +29,9 @@ import os
from morituri.common import log, common
from morituri.image import cue, table
from morituri.extern.task import task, gstreamer
from morituri.extern.task import task
from morituri.program.soxi import AudioLengthTask
class Image(object, log.Loggable):
@@ -146,57 +148,6 @@ class AccurateRipChecksumTask(log.Loggable, task.MultiSeparateTask):
task.MultiSeparateTask.stop(self)
class AudioLengthTask(log.Loggable, gstreamer.GstPipelineTask):
"""
I calculate the length of a track in audio samples.
@ivar length: length of the decoded audio file, in audio samples.
"""
logCategory = 'AudioLengthTask'
description = 'Getting length of audio track'
length = None
playing = False
def __init__(self, path):
"""
@type path: unicode
"""
assert type(path) is unicode, "%r is not unicode" % path
self._path = path
self.logName = os.path.basename(path).encode('utf-8')
def getPipelineDesc(self):
return '''
filesrc location="%s" !
decodebin ! audio/x-raw-int !
fakesink name=sink''' % \
gstreamer.quoteParse(self._path).encode('utf-8')
def paused(self):
self.debug('query duration')
sink = self.pipeline.get_by_name('sink')
assert sink, 'Error constructing pipeline'
try:
length, qformat = sink.query_duration(self.gst.FORMAT_DEFAULT)
except self.gst.QueryError, e:
self.info('failed to query duration of %r' % self._path)
self.setException(e)
raise
# wavparse 0.10.14 returns in bytes
if qformat == self.gst.FORMAT_BYTES:
self.debug('query returned in BYTES format')
length /= 4
self.debug('total length of %r in samples: %d', self._path, length)
self.length = length
self.pipeline.set_state(self.gst.STATE_NULL)
self.stop()
class ImageVerifyTask(log.Loggable, task.MultiSeparateTask):
"""
I verify a disk image and get the necessary track lengths.

47
morituri/program/soxi.py Normal file
View File

@@ -0,0 +1,47 @@
import os
from morituri.common import log, common
from morituri.common import task as ctask
SOXI = 'soxi'
class AudioLengthTask(ctask.PopenTask, log.Loggable):
"""
I calculate the length of a track in audio samples.
@ivar length: length of the decoded audio file, in audio samples.
"""
logCategory = 'AudioLengthTask'
description = 'Getting length of audio track'
length = None
def __init__(self, path):
"""
@type path: unicode
"""
assert type(path) is unicode, "%r is not unicode" % path
self._path = path
self.logName = os.path.basename(path).encode('utf-8')
self.command = [SOXI, '-s', self._path]
self._error = []
self._output = []
def commandMissing(self):
raise common.MissingDependencyException('sox')
def readbytesout(self, bytes):
self._output.append(bytes)
def readbyteserr(self, bytes):
self._error.append(bytes)
def failed(self):
self.setException(Exception("soxi failed: %s"%"".join(self._error)))
def done(self):
if self._error:
self.warning("soxi reported on stderr: %s", "".join(self._error))
self.length = int("".join(self._output))