epicyon/notifications_client.py

439 lines
19 KiB
Python
Raw Normal View History

__filename__ = "notifications_client.py"
2021-03-04 13:57:30 +00:00
__author__ = "Bob Mottram"
__license__ = "AGPL3+"
__version__ = "1.2.0"
__maintainer__ = "Bob Mottram"
__email__ = "bob@freedombone.net"
__status__ = "Production"
import os
import html
import time
2021-03-04 15:21:38 +00:00
import sys
import select
from utils import getNicknameFromActor
from utils import getDomainFromActor
from utils import getFullDomain
2021-03-04 13:57:30 +00:00
from session import createSession
from speaker import getSpeakerFromServer
from speaker import getSpeakerPitch
from speaker import getSpeakerRate
from speaker import getSpeakerRange
2021-03-10 13:04:22 +00:00
from like import sendLikeViaServer
2021-03-10 13:07:24 +00:00
from like import sendUndoLikeViaServer
from follow import sendFollowRequestViaServer
from follow import sendUnfollowRequestViaServer
2021-03-04 14:36:24 +00:00
def _waitForKeypress(timeout: int, debug: bool) -> str:
"""Waits for a keypress with a timeout
Returns the key pressed, or None on timeout
"""
2021-03-04 15:21:38 +00:00
i, o, e = select.select([sys.stdin], [], [], timeout)
if (i):
2021-03-04 15:27:27 +00:00
text = sys.stdin.readline().strip()
2021-03-04 15:21:38 +00:00
if debug:
2021-03-04 15:27:27 +00:00
print("Text entered: " + text)
return text
2021-03-04 15:21:38 +00:00
else:
2021-03-04 14:36:24 +00:00
if debug:
2021-03-04 15:21:38 +00:00
print("Timeout")
2021-03-04 14:36:24 +00:00
return None
2021-03-04 13:57:30 +00:00
2021-03-04 14:54:30 +00:00
def _speakerEspeak(espeak, pitch: int, rate: int, srange: int,
sayText: str) -> None:
"""Speaks the given text with espeak
"""
espeak.set_parameter(espeak.Parameter.Pitch, pitch)
espeak.set_parameter(espeak.Parameter.Rate, rate)
espeak.set_parameter(espeak.Parameter.Range, srange)
espeak.synth(html.unescape(sayText))
def _speakerPicospeaker(pitch: int, rate: int, systemLanguage: str,
sayText: str) -> None:
speakerLang = 'en-GB'
if systemLanguage:
if systemLanguage.startswith('fr'):
speakerLang = 'fr-FR'
elif systemLanguage.startswith('es'):
speakerLang = 'es-ES'
elif systemLanguage.startswith('de'):
speakerLang = 'de-DE'
elif systemLanguage.startswith('it'):
speakerLang = 'it-IT'
speakerCmd = 'picospeaker ' + \
'-l ' + speakerLang + \
' -r ' + str(rate) + \
' -p ' + str(pitch) + ' "' + \
2021-03-10 12:26:10 +00:00
html.unescape(sayText) + '" 2> /dev/null'
2021-03-04 14:54:30 +00:00
os.system(speakerCmd)
2021-03-09 19:52:10 +00:00
def _playNotificationSound(soundFilename: str, player='ffplay') -> None:
"""Plays a sound
"""
if not os.path.isfile(soundFilename):
return
if player == 'ffplay':
os.system('ffplay ' + soundFilename +
2021-03-10 15:43:21 +00:00
' -autoexit -hide_banner -nodisp 2> /dev/null')
2021-03-09 19:52:10 +00:00
2021-03-09 20:32:50 +00:00
def _desktopNotification(notificationType: str,
2021-03-09 21:23:32 +00:00
title: str, message: str) -> None:
2021-03-09 20:32:50 +00:00
"""Shows a desktop notification
"""
if not notificationType:
return
if notificationType == 'notify-send':
# Ubuntu
os.system('notify-send "' + title + '" "' + message + '"')
elif notificationType == 'osascript':
# Mac
os.system("osascript -e 'display notification \"" +
message + "\" with title \"" + title + "\"'")
elif notificationType == 'New-BurntToastNotification':
# Windows
os.system("New-BurntToastNotification -Text \"" +
title + "\", '" + message + "'")
2021-03-10 12:11:42 +00:00
def _textToSpeech(sayStr: str, screenreader: str,
pitch: int, rate: int, srange: int,
systemLanguage: str, espeak=None) -> None:
"""Say something via TTS
"""
# speak the post content
if screenreader == 'espeak':
_speakerEspeak(espeak, pitch, rate, srange, sayStr)
elif screenreader == 'picospeaker':
_speakerPicospeaker(pitch, rate,
systemLanguage, sayStr)
2021-03-10 10:25:41 +00:00
def _sayCommand(sayStr: str, screenreader: str,
2021-03-10 12:11:42 +00:00
systemLanguage: str,
espeak=None,
speakerName='screen reader',
speakerGender='They/Them') -> None:
2021-03-10 10:25:41 +00:00
"""Speaks a command
"""
print(sayStr)
2021-03-10 10:32:08 +00:00
if not screenreader:
return
2021-03-10 10:25:41 +00:00
2021-03-10 12:11:42 +00:00
pitch = getSpeakerPitch(speakerName,
screenreader, speakerGender)
rate = getSpeakerRate(speakerName, screenreader)
srange = getSpeakerRange(speakerName)
2021-03-10 10:25:41 +00:00
2021-03-10 12:11:42 +00:00
_textToSpeech(sayStr, screenreader,
pitch, rate, srange,
systemLanguage, espeak)
2021-03-10 10:25:41 +00:00
def runNotificationsClient(baseDir: str, proxyType: str, httpPrefix: str,
nickname: str, domain: str, port: int,
password: str, screenreader: str,
systemLanguage: str,
notificationSounds: bool,
notificationType: str,
debug: bool) -> None:
"""Runs the notifications and screen reader client,
which announces new inbox items
2021-03-04 13:57:30 +00:00
"""
2021-03-10 10:25:41 +00:00
espeak = None
2021-03-09 19:52:10 +00:00
if screenreader:
if screenreader == 'espeak':
print('Setting up espeak')
from espeak import espeak
elif screenreader != 'picospeaker':
print(screenreader + ' is not a supported TTS system')
return
2021-03-04 13:57:30 +00:00
2021-03-10 10:49:45 +00:00
sayStr = 'Running ' + screenreader + ' for ' + nickname + '@' + domain
_sayCommand(sayStr, screenreader,
systemLanguage, espeak)
2021-03-09 21:30:23 +00:00
else:
print('Running desktop notifications for ' + nickname + '@' + domain)
if notificationSounds:
2021-03-10 10:49:45 +00:00
sayStr = 'Notification sounds on'
else:
2021-03-10 10:49:45 +00:00
sayStr = 'Notification sounds off'
_sayCommand(sayStr, screenreader,
systemLanguage, espeak)
sayStr = '/q or /quit to exit'
_sayCommand(sayStr, screenreader,
systemLanguage, espeak)
2021-03-10 12:41:26 +00:00
print('')
2021-03-10 10:51:06 +00:00
keyPress = _waitForKeypress(2, debug)
2021-03-10 10:25:41 +00:00
originalScreenReader = screenreader
domainFull = getFullDomain(domain, port)
actor = httpPrefix + '://' + domainFull + '/users/' + nickname
2021-03-04 13:57:30 +00:00
prevSay = ''
2021-03-09 19:52:10 +00:00
prevDM = False
prevReply = False
prevCalendar = False
prevFollow = False
prevLike = ''
prevShare = False
dmSoundFilename = 'dm.ogg'
replySoundFilename = 'reply.ogg'
calendarSoundFilename = 'calendar.ogg'
followSoundFilename = 'follow.ogg'
likeSoundFilename = 'like.ogg'
shareSoundFilename = 'share.ogg'
player = 'ffplay'
2021-03-10 12:37:44 +00:00
nameStr = None
gender = None
messageStr = None
2021-03-10 13:04:22 +00:00
cachedWebfingers = {}
personCache = {}
2021-03-04 13:57:30 +00:00
while (1):
session = createSession(proxyType)
speakerJson = \
getSpeakerFromServer(baseDir, session, nickname, password,
domain, port, httpPrefix, True, __version__)
if speakerJson:
2021-03-09 19:52:10 +00:00
if speakerJson.get('notify'):
title = 'Epicyon'
if speakerJson['notify'].get('title'):
title = speakerJson['notify']['title']
2021-03-09 19:52:10 +00:00
soundsDir = 'theme/default/sounds'
if speakerJson['notify'].get('theme'):
2021-03-09 23:22:48 +00:00
if isinstance(speakerJson['notify']['theme'], str):
soundsDir = \
'theme/' + \
speakerJson['notify']['theme'] + '/sounds'
if not os.path.isdir(soundsDir):
soundsDir = 'theme/default/sounds'
2021-03-09 22:47:10 +00:00
if speakerJson['notify']['dm'] != prevDM:
if speakerJson['notify']['dm'] is True:
if notificationSounds:
_playNotificationSound(soundsDir + '/' +
dmSoundFilename, player)
2021-03-09 22:47:10 +00:00
_desktopNotification(notificationType, title,
'New direct message ' +
actor + '/dm')
2021-03-09 22:47:10 +00:00
prevDM = speakerJson['notify']['dm']
elif speakerJson['notify']['reply'] != prevReply:
if speakerJson['notify']['reply'] is True:
if notificationSounds:
_playNotificationSound(soundsDir + '/' +
replySoundFilename,
player)
2021-03-09 22:47:10 +00:00
_desktopNotification(notificationType, title,
'New reply ' +
actor + '/tlreplies')
2021-03-09 22:35:06 +00:00
prevReply = speakerJson['notify']['reply']
2021-03-09 22:47:10 +00:00
elif speakerJson['notify']['calendar'] != prevCalendar:
if speakerJson['notify']['calendar'] is True:
if notificationSounds:
_playNotificationSound(soundsDir + '/' +
calendarSoundFilename,
player)
2021-03-09 22:47:10 +00:00
_desktopNotification(notificationType, title,
'New calendar event ' +
actor + '/calendar')
2021-03-09 22:47:10 +00:00
prevCalendar = speakerJson['notify']['calendar']
elif speakerJson['notify']['followRequests'] != prevFollow:
if speakerJson['notify']['followRequests'] is True:
if notificationSounds:
_playNotificationSound(soundsDir + '/' +
followSoundFilename,
player)
2021-03-09 22:47:10 +00:00
_desktopNotification(notificationType, title,
'New follow request ' +
actor + '/followers#buttonheader')
2021-03-09 22:47:10 +00:00
prevFollow = speakerJson['notify']['followRequests']
elif speakerJson['notify']['likedBy'] != prevLike:
if notificationSounds:
_playNotificationSound(soundsDir + '/' +
likeSoundFilename, player)
2021-03-09 22:47:10 +00:00
_desktopNotification(notificationType, title,
'New like ' +
speakerJson['notify']['likedBy'])
prevLike = speakerJson['notify']['likedBy']
elif speakerJson['notify']['share'] != prevShare:
if speakerJson['notify']['share'] is True:
if notificationSounds:
_playNotificationSound(soundsDir + '/' +
shareSoundFilename,
player)
2021-03-09 22:47:10 +00:00
_desktopNotification(notificationType, title,
'New shared item ' +
actor + '/shares')
2021-03-09 22:47:10 +00:00
prevShare = speakerJson['notify']['share']
2021-03-09 19:52:10 +00:00
if speakerJson.get('say'):
if speakerJson['say'] != prevSay:
if speakerJson.get('name'):
nameStr = speakerJson['name']
gender = 'They/Them'
if speakerJson.get('gender'):
gender = speakerJson['gender']
# append image description if needed
if not speakerJson.get('imageDescription'):
2021-03-10 12:37:44 +00:00
messageStr = speakerJson['say']
2021-03-09 19:52:10 +00:00
else:
2021-03-10 12:37:44 +00:00
messageStr = speakerJson['say'] + '. ' + \
2021-03-09 19:52:10 +00:00
speakerJson['imageDescription']
2021-03-10 12:15:08 +00:00
2021-03-10 12:11:42 +00:00
# say the speaker's name
_sayCommand(nameStr, screenreader,
systemLanguage, espeak,
nameStr, gender)
time.sleep(2)
2021-03-09 19:52:10 +00:00
# speak the post content
2021-03-10 12:37:44 +00:00
_sayCommand(messageStr, screenreader,
2021-03-10 12:11:42 +00:00
systemLanguage, espeak,
nameStr, gender)
2021-03-09 19:52:10 +00:00
2021-03-10 12:40:17 +00:00
print('')
2021-03-09 19:52:10 +00:00
prevSay = speakerJson['say']
2021-03-04 13:57:30 +00:00
# wait for a while, or until a key is pressed
2021-03-04 18:05:46 +00:00
keyPress = _waitForKeypress(30, debug)
if keyPress:
if keyPress.startswith('/'):
keyPress = keyPress[1:]
if keyPress == 'q' or keyPress == 'quit' or keyPress == 'exit':
2021-03-10 10:46:50 +00:00
sayStr = 'Quit'
_sayCommand(sayStr, screenreader,
systemLanguage, espeak)
2021-03-10 12:15:49 +00:00
keyPress = _waitForKeypress(2, debug)
2021-03-04 18:05:46 +00:00
break
2021-03-10 13:04:22 +00:00
elif keyPress == 'like':
if nameStr and gender and messageStr:
_sayCommand('Liking post by ' + nameStr,
screenreader,
systemLanguage, espeak)
sendLikeViaServer(baseDir, session,
nickname, password,
domain, port,
httpPrefix, speakerJson['id'],
cachedWebfingers, personCache,
True, __version__)
2021-03-10 13:13:54 +00:00
print('')
2021-03-10 13:07:24 +00:00
elif keyPress == 'unlike' or keyPress == 'undo like':
if nameStr and gender and messageStr:
_sayCommand('Undoing like of post by ' + nameStr,
screenreader,
systemLanguage, espeak)
sendUndoLikeViaServer(baseDir, session,
nickname, password,
domain, port,
httpPrefix, speakerJson['id'],
cachedWebfingers, personCache,
True, __version__)
2021-03-10 13:13:54 +00:00
print('')
elif keyPress.startswith('follow '):
followHandle = keyPress.replace('follow ', '').strip()
if followHandle.startswith('@'):
followHandle = followHandle[1:]
if '@' in followHandle or '://' in followHandle:
followNickname = getNicknameFromActor(followHandle)
followDomain, followPort = \
getDomainFromActor(followHandle)
if followNickname and followDomain:
_sayCommand('Sending follow request to ' +
followNickname + '@' + followDomain,
screenreader, systemLanguage, espeak)
sendFollowRequestViaServer(baseDir, session,
nickname, password,
domain, port,
followNickname,
followDomain,
followPort,
httpPrefix,
cachedWebfingers,
personCache,
debug, __version__)
else:
_sayCommand(followHandle + ' is not valid',
screenreader, systemLanguage, espeak)
print('')
elif (keyPress.startswith('unfollow ') or
keyPress.startswith('stop following ')):
followHandle = keyPress.replace('unfollow ', '').strip()
followHandle = followHandle.replace('stop following ', '')
if followHandle.startswith('@'):
followHandle = followHandle[1:]
if '@' in followHandle or '://' in followHandle:
followNickname = getNicknameFromActor(followHandle)
followDomain, followPort = \
getDomainFromActor(followHandle)
if followNickname and followDomain:
_sayCommand('Stop following ' +
followNickname + '@' + followDomain,
screenreader, systemLanguage, espeak)
sendUnfollowRequestViaServer(baseDir, session,
nickname, password,
domain, port,
followNickname,
followDomain,
followPort,
httpPrefix,
cachedWebfingers,
personCache,
debug, __version__)
else:
_sayCommand(followHandle + ' is not valid',
screenreader, systemLanguage, espeak)
print('')
2021-03-10 13:13:54 +00:00
elif (keyPress == 'repeat' or keyPress == 'replay' or
keyPress == 'rp'):
2021-03-10 12:37:44 +00:00
if nameStr and gender and messageStr:
_sayCommand('Repeating ' + nameStr, screenreader,
systemLanguage, espeak,
nameStr, gender)
time.sleep(2)
_sayCommand(messageStr, screenreader,
2021-03-10 12:40:17 +00:00
systemLanguage, espeak,
nameStr, gender)
print('')
elif keyPress == 'sounds on' or keyPress == 'sound':
2021-03-10 10:34:06 +00:00
sayStr = 'Notification sounds on'
2021-03-10 10:32:08 +00:00
_sayCommand(sayStr, screenreader,
systemLanguage, espeak)
notificationSounds = True
elif keyPress == 'sounds off' or keyPress == 'nosound':
2021-03-10 10:34:06 +00:00
sayStr = 'Notification sounds off'
2021-03-10 10:32:08 +00:00
_sayCommand(sayStr, screenreader,
systemLanguage, espeak)
notificationSounds = False
2021-03-10 10:32:08 +00:00
elif (keyPress == 'speak' or
keyPress == 'screen reader on' or
keyPress == 'speaker on' or
keyPress == 'talker on' or
keyPress == 'reader on'):
2021-03-10 10:25:41 +00:00
if originalScreenReader:
screenreader = originalScreenReader
2021-03-10 10:34:06 +00:00
sayStr = 'Screen reader on'
2021-03-10 10:25:41 +00:00
_sayCommand(sayStr, screenreader,
systemLanguage, espeak)
else:
print('No --screenreader option was specified')
2021-03-10 10:32:08 +00:00
elif (keyPress == 'mute' or
keyPress == 'screen reader off' or
keyPress == 'speaker off' or
keyPress == 'talker off' or
keyPress == 'reader off'):
2021-03-10 10:25:41 +00:00
if originalScreenReader:
screenreader = None
2021-03-10 10:34:06 +00:00
sayStr = 'Screen reader off'
2021-03-10 10:25:41 +00:00
_sayCommand(sayStr, originalScreenReader,
systemLanguage, espeak)
else:
print('No --screenreader option was specified')