|
@ -137,9 +137,9 @@ class OpusEncoder(BaseOpus): |
|
|
return result |
|
|
return result |
|
|
|
|
|
|
|
|
def __del__(self): |
|
|
def __del__(self): |
|
|
if self.inst: |
|
|
if self._inst: |
|
|
self.opus_encoder_destroy(self.inst) |
|
|
self.opus_encoder_destroy(self._inst) |
|
|
self.inst = None |
|
|
self._inst = None |
|
|
|
|
|
|
|
|
def encode(self, pcm, frame_size): |
|
|
def encode(self, pcm, frame_size): |
|
|
max_data_bytes = len(pcm) |
|
|
max_data_bytes = len(pcm) |
|
@ -159,24 +159,24 @@ class OpusDecoder(BaseOpus): |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class BufferedOpusEncoder(OpusEncoder): |
|
|
class BufferedOpusEncoder(OpusEncoder): |
|
|
def __init__(self, f, *args, **kwargs): |
|
|
def __init__(self, source, *args, **kwargs): |
|
|
self.data = f |
|
|
self.source = source |
|
|
self.frames = Queue(kwargs.pop('queue_size', 4096)) |
|
|
self.frames = Queue(kwargs.pop('queue_size', 4096)) |
|
|
super(BufferedOpusEncoder, self).__init__(*args, **kwargs) |
|
|
super(BufferedOpusEncoder, self).__init__(*args, **kwargs) |
|
|
gevent.spawn(self._encoder_loop) |
|
|
gevent.spawn(self._encoder_loop) |
|
|
|
|
|
|
|
|
def _encoder_loop(self): |
|
|
def _encoder_loop(self): |
|
|
while self.data: |
|
|
while self.source: |
|
|
raw = self.data.read(self.frame_size) |
|
|
raw = self.source.read(self.frame_size) |
|
|
if len(raw) < self.frame_size: |
|
|
if len(raw) < self.frame_size: |
|
|
break |
|
|
break |
|
|
|
|
|
|
|
|
self.frames.put(self.encode(raw, self.samples_per_frame)) |
|
|
self.frames.put(self.encode(raw, self.samples_per_frame)) |
|
|
gevent.idle() |
|
|
gevent.idle() |
|
|
self.data = None |
|
|
self.source = None |
|
|
|
|
|
|
|
|
def have_frame(self): |
|
|
def have_frame(self): |
|
|
return self.data or not self.frames.empty() |
|
|
return self.source or not self.frames.empty() |
|
|
|
|
|
|
|
|
def next_frame(self): |
|
|
def next_frame(self): |
|
|
return self.frames.get() |
|
|
return self.frames.get() |
|
@ -185,10 +185,10 @@ class BufferedOpusEncoder(OpusEncoder): |
|
|
class GIPCBufferedOpusEncoder(OpusEncoder): |
|
|
class GIPCBufferedOpusEncoder(OpusEncoder): |
|
|
FIN = 1 |
|
|
FIN = 1 |
|
|
|
|
|
|
|
|
def __init__(self, f, *args, **kwargs): |
|
|
def __init__(self, source, *args, **kwargs): |
|
|
import gipc |
|
|
import gipc |
|
|
|
|
|
|
|
|
self.data = f |
|
|
self.source = source |
|
|
self.parent_pipe, self.child_pipe = gipc.pipe(duplex=True) |
|
|
self.parent_pipe, self.child_pipe = gipc.pipe(duplex=True) |
|
|
self.frames = Queue(kwargs.pop('queue_size', 4096)) |
|
|
self.frames = Queue(kwargs.pop('queue_size', 4096)) |
|
|
super(GIPCBufferedOpusEncoder, self).__init__(*args, **kwargs) |
|
|
super(GIPCBufferedOpusEncoder, self).__init__(*args, **kwargs) |
|
@ -209,7 +209,7 @@ class GIPCBufferedOpusEncoder(OpusEncoder): |
|
|
|
|
|
|
|
|
def _writer(self): |
|
|
def _writer(self): |
|
|
while self.data: |
|
|
while self.data: |
|
|
raw = self.data.read(self.frame_size) |
|
|
raw = self.source.read(self.frame_size) |
|
|
if len(raw) < self.frame_size: |
|
|
if len(raw) < self.frame_size: |
|
|
break |
|
|
break |
|
|
|
|
|
|
|
@ -238,36 +238,48 @@ class GIPCBufferedOpusEncoder(OpusEncoder): |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class DCADOpusEncoder(OpusEncoder): |
|
|
class DCADOpusEncoder(OpusEncoder): |
|
|
def __init__(self, pipe, *args, **kwargs): |
|
|
def __init__(self, source, *args, **kwargs): |
|
|
command = kwargs.pop('command', 'dcad') |
|
|
self.source = source |
|
|
|
|
|
self.command = kwargs.pop('command', 'dcad') |
|
|
super(DCADOpusEncoder, self).__init__(*args, **kwargs) |
|
|
super(DCADOpusEncoder, self).__init__(*args, **kwargs) |
|
|
self.proc = subprocess.Popen([ |
|
|
self._proc = None |
|
|
command, |
|
|
|
|
|
# '--channels', str(self.channels), |
|
|
|
|
|
# '--rate', str(self.sampling_rate), |
|
|
|
|
|
# '--size', str(self.frame_length), |
|
|
|
|
|
'--bitrate', '128', |
|
|
|
|
|
'--fec', |
|
|
|
|
|
'--packet-loss-percent', '30', |
|
|
|
|
|
'--input', 'pipe:0', |
|
|
|
|
|
'--output', 'pipe:1', |
|
|
|
|
|
], stdin=pipe, stdout=subprocess.PIPE) |
|
|
|
|
|
self.header_size = struct.calcsize('<h') |
|
|
self.header_size = struct.calcsize('<h') |
|
|
|
|
|
|
|
|
|
|
|
@property |
|
|
|
|
|
def proc(self): |
|
|
|
|
|
if not self._proc: |
|
|
|
|
|
self._proc = subprocess.Popen([ |
|
|
|
|
|
self.command, |
|
|
|
|
|
'--channels', str(self.channels), |
|
|
|
|
|
'--rate', str(self.sampling_rate), |
|
|
|
|
|
'--size', str(self.samples_per_frame), |
|
|
|
|
|
'--bitrate', '128', |
|
|
|
|
|
'--fec', |
|
|
|
|
|
'--packet-loss-percent', '30', |
|
|
|
|
|
'--input', 'pipe:0', |
|
|
|
|
|
'--output', 'pipe:1', |
|
|
|
|
|
], stdin=self.source.stdout, stdout=subprocess.PIPE, stderr=subprocess.PIPE, close_fds=True) |
|
|
|
|
|
return self._proc |
|
|
|
|
|
|
|
|
def have_frame(self): |
|
|
def have_frame(self): |
|
|
return bool(self.proc) |
|
|
return bool(self.proc) |
|
|
|
|
|
|
|
|
def next_frame(self): |
|
|
def next_frame(self): |
|
|
header = self.proc.stdout.read(self.header_size) |
|
|
header = self.proc.stdout.read(self.header_size) |
|
|
if len(header) < self.header_size: |
|
|
if len(header) < self.header_size: |
|
|
self.proc = None |
|
|
print 'read less than required header size' |
|
|
|
|
|
print self.proc.poll() |
|
|
|
|
|
self._proc = None |
|
|
return |
|
|
return |
|
|
|
|
|
|
|
|
|
|
|
if self.proc.poll() is not None: |
|
|
|
|
|
print 'read that data when she dead n gone: %s' % self.proc.poll() |
|
|
|
|
|
|
|
|
size = struct.unpack('<h', header)[0] |
|
|
size = struct.unpack('<h', header)[0] |
|
|
|
|
|
|
|
|
data = self.proc.stdout.read(size) |
|
|
data = self.proc.stdout.read(size) |
|
|
if len(data) < size: |
|
|
if len(data) == 0: |
|
|
self.proc = None |
|
|
self._proc = None |
|
|
return |
|
|
return |
|
|
|
|
|
|
|
|
return data |
|
|
return data |
|
|