You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
226 lines
7.7 KiB
226 lines
7.7 KiB
import linphone |
|
import time |
|
import threading |
|
import subprocess |
|
from pylinphone import LinphoneCommunicationSocket |
|
|
|
|
|
class PhoneProxyConfiguration(object): |
|
def __init__(self, name, proxy, identity, username, password, realm, |
|
prefix): |
|
self.name = name |
|
self.proxy = proxy |
|
self.identity = identity |
|
self.username = username |
|
self.password = password |
|
self.realm = realm |
|
self.prefix = prefix |
|
|
|
|
|
class PhoneConfiguration(object): |
|
def __init__(self, sound_device, incoming_timeout, linphone_config, |
|
default_proxy, proxies, stun_server): |
|
self.sound_device = sound_device |
|
self.incoming_timeout = incoming_timeout |
|
self.linphone_config = linphone_config |
|
self.default_proxy = default_proxy |
|
self.proxies = proxies |
|
self.stun_server = stun_server |
|
|
|
|
|
class PhoneEvent(object): |
|
RegInProgress,\ |
|
RegSuccessfull,\ |
|
RegLost,\ |
|
CallIncoming,\ |
|
CallRinging,\ |
|
CallAccepted,\ |
|
CallEnded,\ |
|
CallBusy,\ |
|
CallInvalidNumber = range(9) |
|
|
|
@classmethod |
|
def string(cls, val): |
|
for k, v in vars(cls).items(): |
|
if v == val: |
|
return k |
|
|
|
|
|
class PhoneInterface(object): |
|
def __init__(self, config): |
|
cbs = { |
|
'global_state_changed': self.__global_state_changed, |
|
'registration_state_changed': self.__registration_state_changed, |
|
'call_state_changed': self.__call_state_changed |
|
} |
|
|
|
self.__event_cbs = [] |
|
|
|
self.__config = config |
|
self.__core = LinphoneCommunicationSocket("/tmp/lpdaemon") |
|
|
|
# Create and add all proxy configs |
|
for p in config.proxies: |
|
ainfo = self.__core.create_auth_info(p.username, p.username, |
|
p.password, None, p.realm, |
|
None) |
|
aid = self.__core.register(p.username, p.proxy_address, p.password, p.username) # sip:XXXX@hg.eventphone.de, hg.eventphone.de, MySecretPassword, XXXX |
|
|
|
self.__audioproc = None |
|
aplay = subprocess.Popen(['aplay', '-qD%s' % config.sound_device], |
|
stdin=subprocess.PIPE) |
|
self.__ttsproc = subprocess.Popen(['espeak', '-p10', '--stdout'], |
|
stdin=subprocess.PIPE, |
|
stdout=aplay.stdin) |
|
|
|
# Set default parameters overriding the ones from the given config file |
|
# TODO: figure out how to set at least some of these settings through the unix socket |
|
#self.__core.set_user_agent('FeTAp 615', '0.1') |
|
#self.__core.stun_server = config.stun_server |
|
#self.__core.ringback = '' |
|
#self.__core.max_calls = 1 |
|
#self.__core.inc_timeout = config.incoming_timeout |
|
#self.__core.set_call_error_tone(linphone.Reason.Busy, '') |
|
#self.__core.disable_chat(linphone.Reason.None) |
|
#self.__core.echo_cancellation_enabled = False |
|
#self.__core.video_capture_enabled = False |
|
#self.__core.video_display_enabled = False |
|
|
|
def __global_state_changed(self, core, state, msg): |
|
print('Global state changed:', state, msg) |
|
# TODO: Do we need events emitted here? |
|
pass |
|
|
|
def __registration_state_changed(self, core, proxyconf, state, msg): |
|
print('Registration state changed:', proxyconf, state, msg) |
|
evt = None |
|
if state == linphone.RegistrationState.Progress: |
|
evt = PhoneEvent.RegInProgress |
|
elif state == linphone.RegistrationState.Ok: |
|
evt = PhoneEvent.RegSuccessfull |
|
elif state == linphone.RegistrationState.None: |
|
evt = PhoneEvent.RegLost |
|
|
|
if evt is not None: |
|
for cb in self.__event_cbs: |
|
cb(evt) |
|
else: |
|
print('Unhandled registration state:', linphone.RegistrationState.string(state)) |
|
|
|
def __call_state_changed(self, core, call, state, msg): |
|
print('Call state changed:', call, state, msg) |
|
evt = None |
|
if state == linphone.CallState.IncomingReceived: |
|
evt = PhoneEvent.CallIncoming |
|
elif state == linphone.CallState.OutgoingRinging: |
|
evt = PhoneEvent.CallRinging |
|
elif state == linphone.CallState.Connected: |
|
evt = PhoneEvent.CallAccepted |
|
elif state == linphone.CallState.End: |
|
evt = PhoneEvent.CallEnded |
|
elif state == linphone.CallState.Error: |
|
error = call.error_info.reason |
|
if error == linphone.Reason.Busy: |
|
evt = PhoneEvent.CallBusy |
|
elif error == linphone.Reason.NotFound: |
|
evt = PhoneEvent.CallInvalidNumber |
|
else: |
|
evt = PhoneEvent.CallEnded |
|
|
|
if evt is not None: |
|
for cb in self.__event_cbs: |
|
cb(evt) |
|
else: |
|
print('Unhandled call state:', linphone.CallState.string(state)) |
|
|
|
def __pollthread(self): |
|
while self.__running: |
|
self.__core.process_event() |
|
time.sleep(0.2) # Value for good measure |
|
|
|
def start(self): |
|
self.__running = True |
|
t = threading.Thread(target=self.__pollthread) |
|
t.start() |
|
|
|
def stop(self): |
|
self.stop_playing() |
|
if self.__ttsproc is not None: |
|
self.__ttsproc.terminate() |
|
self.__running = False |
|
|
|
def add_event_cb(self, cb): |
|
self.__event_cbs.append(cb) |
|
|
|
def call(self, number): |
|
if '@' not in number and self.__core.default_proxy_config is None: |
|
# Try to resolve prefix |
|
for p in self.__config.proxies: |
|
if number.startswith(p.prefix): |
|
number = number[len(p.prefix):] |
|
number += '@' + p.realm |
|
break |
|
self.__core.invite(number) |
|
|
|
def accept_call(self): |
|
self.__core.accept_call(self.__core.current_call) |
|
|
|
def decline_call(self): |
|
self.__core.decline_call(self.__core.current_call, linphone.Reason.Busy) |
|
|
|
def end_call(self): |
|
self.__core.terminate_call(self.__core.current_call) |
|
|
|
def play_dial_tone(self): |
|
self.stop_playing() |
|
self.__audioproc = subprocess.Popen( |
|
['play', '-nq', 'synth', 'sine', '425'], |
|
env={'AUDIODRIVER': 'alsa', 'AUDIODEV': self.__config.sound_device} |
|
) |
|
|
|
def play_ringback_tone(self): |
|
self.stop_playing() |
|
self.__audioproc = subprocess.Popen( |
|
['play', '-nq', 'synth', '1', 'sine', '425', 'pad', '4@1', |
|
'repeat', '1000'], |
|
env={'AUDIODRIVER': 'alsa', 'AUDIODEV': self.__config.sound_device} |
|
) |
|
|
|
def play_busy_tone(self): |
|
self.stop_playing() |
|
self.__audioproc = subprocess.Popen( |
|
['play', '-nq', 'synth', '0.48', 'sine', '425', 'pad', '0.48@0.48', |
|
'repeat', '1000'], |
|
env={'AUDIODRIVER': 'alsa', 'AUDIODEV': self.__config.sound_device} |
|
) |
|
|
|
def stop_playing(self): |
|
if self.__audioproc is not None: |
|
self.__audioproc.terminate() |
|
|
|
def read_text(self, text): |
|
self.__ttsproc.stdin.write(text.encode('utf8') + b'\n') |
|
self.__ttsproc.stdin.flush() |
|
|
|
def get_remote_number(self): |
|
return self.__core.current_call_remote_address.username |
|
|
|
if __name__ == '__main__': |
|
def event_cb(evt): |
|
print('Got event:', PhoneEvent.string(evt)) |
|
|
|
try: |
|
phone = PhoneInterface('.linphonerc-foo', '.linphonerc') |
|
phone.add_event_cb(event_cb) |
|
phone.start() |
|
i = 0 |
|
while True: |
|
time.sleep(1) |
|
i += 1 |
|
if i == 5: |
|
phone.call('3474') |
|
#phone.play_busy_tone() |
|
pass |
|
except KeyboardInterrupt: |
|
phone.stop() |
|
|
|
|