2017-02-01 01:20:17 +01:00
|
|
|
# Safe Eyes is a utility to remind you to take break frequently
|
|
|
|
# to protect your eyes from eye strain.
|
|
|
|
|
|
|
|
# Copyright (C) 2017 Gobinath
|
|
|
|
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2017-02-06 03:09:04 +01:00
|
|
|
import gi
|
|
|
|
gi.require_version('Gdk', '3.0')
|
|
|
|
from gi.repository import Gdk, GLib
|
2017-04-19 01:34:22 +02:00
|
|
|
from html.parser import HTMLParser
|
2017-04-19 19:56:06 +02:00
|
|
|
import babel.dates, os, errno, re, subprocess, threading, logging, locale, json, shutil, pyaudio, wave
|
2017-02-01 01:20:17 +01:00
|
|
|
|
2017-02-12 14:36:09 +01:00
|
|
|
bin_directory = os.path.dirname(os.path.realpath(__file__))
|
|
|
|
home_directory = os.path.expanduser('~')
|
2017-04-10 17:16:02 +02:00
|
|
|
system_language_directory = os.path.join(bin_directory, 'config/lang')
|
2017-04-07 22:20:23 +02:00
|
|
|
config_directory = os.path.join(home_directory, '.config/safeeyes')
|
2017-02-12 14:36:09 +01:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
|
2017-02-01 01:20:17 +01:00
|
|
|
def play_notification():
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Play the alert.wav
|
|
|
|
"""
|
2017-04-10 17:16:02 +02:00
|
|
|
logging.info('Playing audible alert')
|
2017-02-20 20:17:50 +01:00
|
|
|
CHUNK = 1024
|
|
|
|
|
2017-02-01 01:20:17 +01:00
|
|
|
try:
|
2017-02-20 20:17:50 +01:00
|
|
|
# Open the sound file
|
2017-04-07 22:20:23 +02:00
|
|
|
path = get_resource_path('alert.wav')
|
|
|
|
if path is None:
|
|
|
|
return
|
2017-02-20 20:17:50 +01:00
|
|
|
sound = wave.open(path, 'rb')
|
|
|
|
|
|
|
|
# Create a sound stream
|
|
|
|
wrapper = pyaudio.PyAudio()
|
|
|
|
stream = wrapper.open(format=wrapper.get_format_from_width(sound.getsampwidth()),
|
2017-04-19 01:34:22 +02:00
|
|
|
channels=sound.getnchannels(),
|
|
|
|
rate=sound.getframerate(),
|
|
|
|
output=True)
|
2017-02-20 20:17:50 +01:00
|
|
|
|
|
|
|
# Write file data into the sound stream
|
|
|
|
data = sound.readframes(CHUNK)
|
2017-03-23 01:07:51 +01:00
|
|
|
while data != b'':
|
2017-02-20 20:17:50 +01:00
|
|
|
stream.write(data)
|
|
|
|
data = sound.readframes(CHUNK)
|
|
|
|
|
|
|
|
# Close steam
|
|
|
|
stream.stop_stream()
|
|
|
|
stream.close()
|
2017-03-23 01:07:51 +01:00
|
|
|
sound.close()
|
2017-02-20 20:17:50 +01:00
|
|
|
wrapper.terminate()
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
logging.warning('Unable to play audible alert')
|
|
|
|
logging.exception(e)
|
2017-02-06 03:09:04 +01:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
|
|
|
|
def get_resource_path(resource_name):
|
|
|
|
"""
|
2017-04-07 22:20:23 +02:00
|
|
|
Return the user-defined resource if a system resource is overridden by the user.
|
|
|
|
Otherwise, return the system resource. Return None if the specified resource does not exist.
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-04-07 22:20:23 +02:00
|
|
|
if resource_name is None:
|
|
|
|
return None
|
|
|
|
resource_location = os.path.join(config_directory, 'resource', resource_name)
|
|
|
|
if not os.path.isfile(resource_location):
|
|
|
|
resource_location = os.path.join(bin_directory, 'resource', resource_name)
|
|
|
|
if not os.path.isfile(resource_location):
|
|
|
|
logging.error('Resource not found: ' + resource_name)
|
|
|
|
resource_location = None
|
|
|
|
|
|
|
|
return resource_location
|
|
|
|
|
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def system_idle_time():
|
|
|
|
"""
|
2017-02-06 03:09:04 +01:00
|
|
|
Get system idle time in minutes.
|
|
|
|
Return the idle time if xprintidle is available, otherwise return 0.
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-02-06 03:09:04 +01:00
|
|
|
try:
|
|
|
|
return int(subprocess.check_output(['xprintidle']).decode('utf-8')) / 60000 # Convert to minutes
|
|
|
|
except:
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
2017-02-16 00:29:14 +01:00
|
|
|
def start_thread(target_function, **args):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Execute the function in a separate thread.
|
|
|
|
"""
|
2017-02-09 03:39:27 +01:00
|
|
|
thread = threading.Thread(target=target_function, kwargs=args)
|
2017-02-06 03:09:04 +01:00
|
|
|
thread.start()
|
|
|
|
|
|
|
|
|
2017-02-10 02:06:04 +01:00
|
|
|
def execute_main_thread(target_function, args=None):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Execute the given function in main thread.
|
|
|
|
"""
|
2017-02-10 02:06:04 +01:00
|
|
|
if args:
|
|
|
|
GLib.idle_add(lambda: target_function(args))
|
|
|
|
else:
|
|
|
|
GLib.idle_add(lambda: target_function())
|
2017-02-06 03:09:04 +01:00
|
|
|
|
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def is_active_window_skipped(skip_break_window_classes, take_break_window_classes, unfullscreen_allowed=False):
|
|
|
|
"""
|
2017-02-09 03:39:27 +01:00
|
|
|
Check for full-screen applications.
|
2017-02-16 00:29:14 +01:00
|
|
|
This method must be executed by the main thread. If not, it will cause to random failure.
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-04-10 17:16:02 +02:00
|
|
|
logging.info('Searching for full-screen application')
|
2017-02-06 03:09:04 +01:00
|
|
|
screen = Gdk.Screen.get_default()
|
2017-03-13 14:30:44 +01:00
|
|
|
|
2017-04-07 22:20:23 +02:00
|
|
|
active_window = screen.get_active_window()
|
|
|
|
if active_window:
|
|
|
|
active_xid = str(active_window.get_xid())
|
|
|
|
cmdlist = ['xprop', '-root', '-notype','-id',active_xid, 'WM_CLASS', '_NET_WM_STATE']
|
|
|
|
|
|
|
|
try:
|
|
|
|
stdout = subprocess.check_output(cmdlist).decode('utf-8')
|
|
|
|
except subprocess.CalledProcessError:
|
2017-04-10 17:16:02 +02:00
|
|
|
logging.warning('Error in finding full-screen application')
|
2017-04-07 22:20:23 +02:00
|
|
|
pass
|
|
|
|
else:
|
|
|
|
if stdout:
|
|
|
|
is_fullscreen = 'FULLSCREEN' in stdout
|
|
|
|
# Extract the process name
|
|
|
|
process_names = re.findall('"(.+?)"', stdout)
|
|
|
|
if process_names:
|
2017-04-19 01:34:22 +02:00
|
|
|
process = process_names[1].lower()
|
|
|
|
if process in skip_break_window_classes:
|
|
|
|
return True
|
|
|
|
elif process in take_break_window_classes:
|
|
|
|
if is_fullscreen and unfullscreen_allowed:
|
|
|
|
try:
|
|
|
|
active_window.unfullscreen()
|
|
|
|
except:
|
|
|
|
logging.error('Error in unfullscreen the window ' + process)
|
|
|
|
pass
|
|
|
|
return False
|
2017-04-07 22:20:23 +02:00
|
|
|
|
|
|
|
return is_fullscreen
|
|
|
|
|
|
|
|
return False
|
2017-02-09 03:39:27 +01:00
|
|
|
|
|
|
|
|
2017-03-13 14:30:44 +01:00
|
|
|
def __system_locale():
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Return the system locale. If not available, return en_US.UTF-8.
|
|
|
|
"""
|
2017-02-13 15:55:08 +01:00
|
|
|
locale.setlocale(locale.LC_ALL, '')
|
|
|
|
system_locale = locale.getlocale(locale.LC_TIME)[0]
|
2017-02-09 03:39:27 +01:00
|
|
|
if not system_locale:
|
|
|
|
system_locale = 'en_US.UTF-8'
|
2017-03-13 14:30:44 +01:00
|
|
|
return system_locale
|
|
|
|
|
|
|
|
|
|
|
|
def format_time(time):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Format time based on the system time.
|
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
system_locale = __system_locale()
|
2017-02-09 03:39:27 +01:00
|
|
|
return babel.dates.format_time(time, format='short', locale=system_locale)
|
2017-02-10 02:45:49 +01:00
|
|
|
|
|
|
|
|
|
|
|
def mkdir(path):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Create directory if not exists.
|
|
|
|
"""
|
2017-02-10 02:45:49 +01:00
|
|
|
try:
|
|
|
|
os.makedirs(path)
|
|
|
|
except OSError as exc:
|
|
|
|
if exc.errno == errno.EEXIST and os.path.isdir(path):
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
logging.error('Error while creating ' + str(path))
|
2017-02-12 17:44:48 +01:00
|
|
|
raise
|
2017-03-13 14:30:44 +01:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def parse_language_code(lang_code):
|
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
Convert the user defined language code to a valid one.
|
|
|
|
This includes converting to lower case and finding system locale language,
|
|
|
|
if the given lang_code code is 'system'.
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
# Convert to lower case
|
|
|
|
lang_code = str(lang_code).lower()
|
|
|
|
|
|
|
|
# If it is system, use the system language
|
|
|
|
if lang_code == 'system':
|
|
|
|
logging.info('Use system language for Safe Eyes')
|
|
|
|
system_locale = __system_locale()
|
|
|
|
lang_code = system_locale[0:2].lower()
|
|
|
|
|
|
|
|
# Check whether translation is available for this language.
|
|
|
|
# If not available, use English by default.
|
|
|
|
language_file_path = os.path.join(system_language_directory, lang_code + '.json')
|
|
|
|
if not os.path.exists(language_file_path):
|
|
|
|
logging.warn('The language {} does not exist. Use English instead'.format(lang_code))
|
|
|
|
lang_code = 'en'
|
|
|
|
|
|
|
|
return lang_code
|
|
|
|
|
|
|
|
|
|
|
|
def load_language(lang_code):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Load the desired language from the available list based on the preference.
|
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
# Convert the user defined language code to a valid one
|
|
|
|
lang_code = parse_language_code(lang_code)
|
|
|
|
|
|
|
|
# Construct the translation file path
|
|
|
|
language_file_path = os.path.join(system_language_directory, lang_code + '.json')
|
|
|
|
|
|
|
|
language = None
|
|
|
|
# Read the language file and construct the json object
|
|
|
|
with open(language_file_path) as language_file:
|
|
|
|
language = json.load(language_file)
|
|
|
|
|
|
|
|
return language
|
|
|
|
|
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def read_lang_files():
|
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
Read all the language translations and build a key-value mapping of language names
|
|
|
|
in English and ISO 639-1 (Filename without extension).
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
2017-03-13 14:30:44 +01:00
|
|
|
languages = {}
|
|
|
|
for lang_file_name in os.listdir(system_language_directory):
|
|
|
|
lang_file_path = os.path.join(system_language_directory, lang_file_name)
|
|
|
|
if os.path.isfile(lang_file_path):
|
|
|
|
with open(lang_file_path) as lang_file:
|
|
|
|
lang = json.load(lang_file)
|
|
|
|
languages[lang_file_name.lower().replace('.json', '')] = lang['meta_info']['language_name']
|
|
|
|
|
|
|
|
return languages
|
2017-04-08 15:30:44 +02:00
|
|
|
|
2017-04-09 02:29:51 +02:00
|
|
|
def lock_screen_command():
|
2017-04-08 15:30:44 +02:00
|
|
|
"""
|
2017-04-09 02:29:51 +02:00
|
|
|
Function tries to detect the screensaver command based on the current envinroment
|
|
|
|
Possible results:
|
2017-04-13 15:33:42 +02:00
|
|
|
Gnome, Unity, Budgie: ['gnome-screensaver-command', '--lock']
|
|
|
|
Cinnamon: ['cinnamon-screensaver-command', '--lock']
|
|
|
|
Pantheon, LXDE: ['light-locker-command', '--lock']
|
|
|
|
Mate: ['mate-screensaver-command', '--lock']
|
|
|
|
KDE: ['qdbus', 'org.freedesktop.ScreenSaver', '/ScreenSaver', 'Lock']
|
|
|
|
XFCE: ['xflock4']
|
|
|
|
Otherwise: None
|
2017-04-08 15:30:44 +02:00
|
|
|
"""
|
2017-04-10 17:16:02 +02:00
|
|
|
desktop_session = os.environ.get('DESKTOP_SESSION')
|
2017-04-20 15:43:26 +02:00
|
|
|
current_desktop = os.environ.get('XDG_CURRENT_DESKTOP')
|
2017-04-09 02:29:51 +02:00
|
|
|
if desktop_session is not None:
|
|
|
|
desktop_session = desktop_session.lower()
|
2017-04-20 15:43:26 +02:00
|
|
|
if ('xfce' in desktop_session or desktop_session.startswith('xubuntu') or (current_desktop is not None and 'xfce' in current_desktop)) and command_exist('xflock4'):
|
|
|
|
return ['xflock4']
|
|
|
|
elif desktop_session == 'cinnamon' and command_exist('cinnamon-screensaver-command'):
|
2017-04-10 17:16:02 +02:00
|
|
|
return ['cinnamon-screensaver-command', '--lock']
|
2017-04-20 15:43:26 +02:00
|
|
|
elif (desktop_session == 'pantheon' or desktop_session.startswith('lubuntu')) and command_exist('light-locker-command'):
|
2017-04-12 17:00:28 +02:00
|
|
|
return ['light-locker-command', '--lock']
|
2017-04-20 15:43:26 +02:00
|
|
|
elif desktop_session == 'mate' and command_exist('mate-screensaver-command'):
|
2017-04-10 17:16:02 +02:00
|
|
|
return ['mate-screensaver-command', '--lock']
|
|
|
|
elif desktop_session == 'kde' or 'plasma' in desktop_session or desktop_session.startswith('kubuntu') or os.environ.get('KDE_FULL_SESSION') == 'true':
|
|
|
|
return ['qdbus', 'org.freedesktop.ScreenSaver', '/ScreenSaver', 'Lock']
|
2017-04-20 15:43:26 +02:00
|
|
|
elif desktop_session in ['gnome','unity', 'budgie-desktop'] or desktop_session.startswith('ubuntu'):
|
|
|
|
if command_exist('gnome-screensaver-command'):
|
|
|
|
return ['gnome-screensaver-command', '--lock']
|
|
|
|
else:
|
|
|
|
# From Gnome 3.8 no gnome-screensaver-command
|
|
|
|
return ['dbus-send', '--type=method_call', '--dest=org.gnome.ScreenSaver', '/org/gnome/ScreenSaver', 'org.gnome.ScreenSaver.Lock']
|
2017-04-09 02:29:51 +02:00
|
|
|
elif os.environ.get('GNOME_DESKTOP_SESSION_ID'):
|
2017-04-20 15:43:26 +02:00
|
|
|
if not 'deprecated' in os.environ.get('GNOME_DESKTOP_SESSION_ID') and command_exist('gnome-screensaver-command'):
|
|
|
|
# Gnome 2
|
2017-04-10 17:16:02 +02:00
|
|
|
return ['gnome-screensaver-command', '--lock']
|
2017-04-08 15:30:44 +02:00
|
|
|
return None
|
|
|
|
|
|
|
|
|
2017-04-20 15:43:26 +02:00
|
|
|
def lock_desktop(command):
|
2017-04-09 02:29:51 +02:00
|
|
|
"""
|
|
|
|
Lock the screen using the predefined commands
|
|
|
|
"""
|
2017-04-20 15:43:26 +02:00
|
|
|
if command:
|
2017-04-09 02:29:51 +02:00
|
|
|
try:
|
|
|
|
subprocess.Popen(command)
|
|
|
|
except Exception as e:
|
2017-04-19 19:56:06 +02:00
|
|
|
logging.error('Error in executing the commad' + str(command) + ' to lock screen')
|
2017-04-19 01:34:22 +02:00
|
|
|
|
|
|
|
|
|
|
|
def html_to_text(html):
|
|
|
|
"""
|
|
|
|
Convert HTML to plain text
|
|
|
|
"""
|
|
|
|
extractor = __HTMLTextExtractor()
|
|
|
|
extractor.feed(html)
|
|
|
|
return extractor.get_data()
|
|
|
|
|
|
|
|
|
2017-04-19 19:56:06 +02:00
|
|
|
def command_exist(command):
|
|
|
|
"""
|
|
|
|
Check whether the given command exist in the system or not.
|
|
|
|
"""
|
|
|
|
if shutil.which(command):
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2017-04-19 01:34:22 +02:00
|
|
|
class __HTMLTextExtractor(HTMLParser):
|
|
|
|
"""
|
|
|
|
Helper class to convert HTML to text
|
|
|
|
"""
|
|
|
|
def __init__(self):
|
|
|
|
self.reset()
|
|
|
|
self.strict = False
|
|
|
|
self.convert_charrefs= True
|
|
|
|
self.fed = []
|
|
|
|
|
|
|
|
def handle_data(self, d):
|
|
|
|
self.fed.append(d)
|
|
|
|
|
|
|
|
def get_data(self):
|
2017-04-20 15:43:26 +02:00
|
|
|
return ''.join(self.fed)
|