# -*- coding: utf-8 -*- # # Greeter.py # # Copyright © 2017 Antergos # Copyright © 2021 JezerM # # This file is part of Web Greeter. # # Web Greeter 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. # # Web Greeter 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. # # The following additional terms are in effect as per Section 7 of the license: # # The preservation of all legal notices and author attributions in # the material or in the Appropriate Legal Notices displayed # by works containing it is required. # # You should have received a copy of the GNU General Public License # along with Web Greeter; If not, see . # Standard Lib from browser.error_prompt import Dialog import subprocess import threading # 3rd-Party Libs import gi gi.require_version('LightDM', '1') from gi.repository import LightDM from browser.bridge import Bridge, BridgeObject from PyQt5.QtCore import QFileSystemWatcher, QVariant, QTimer from config import web_greeter_config from utils.battery import Battery from utils.screensaver import reset_screensaver import globals # This Application from . import ( language_to_dict, layout_to_dict, session_to_dict, user_to_dict, battery_to_dict, logger ) # import utils.battery as battery LightDMGreeter = LightDM.Greeter() LightDMUsers = LightDM.UserList() def changeBrightness(method: str, quantity: int = None): backlight = web_greeter_config["config"]["features"]["backlight"] if not backlight["enabled"]: return if not quantity: quantity = backlight["value"] try: steps = backlight["steps"] child = subprocess.run(["xbacklight", method, str(quantity), "-steps", str(steps)]) if child.returncode == 1: raise ChildProcessError("xbacklight returned 1") except Exception as err: logger.error("Brightness: {}".format(err)) else: if globals.greeter: globals.greeter.greeter.brightness_update.emit() def increaseBrightness(quantity: int = None): backlight = web_greeter_config["config"]["features"]["backlight"] if not backlight["enabled"]: return if not quantity: quantity = backlight["value"] thread = threading.Thread(target=changeBrightness, args=("-inc", quantity)) thread.start() def decreaseBrightness(quantity: int = None): backlight = web_greeter_config["config"]["features"]["backlight"] if not backlight["enabled"]: return if not quantity: quantity = backlight["value"] thread = threading.Thread(target=changeBrightness, args=("-dec", quantity)) thread.start() def setBrightness(quantity: int = None): backlight = web_greeter_config["config"]["features"]["backlight"] if not backlight["enabled"]: return if not quantity: quantity = backlight["value"] thread = threading.Thread(target=changeBrightness, args=("-set", quantity)) thread.start() def getBrightness(self): if self._config["features"]["backlight"]["enabled"] != True: return -1 try: level = subprocess.run(["xbacklight", "-get"], stdout=subprocess.PIPE, stderr=subprocess.PIPE, text=True, check=True) return int(level.stdout) except Exception as err: logger.error("Brightness: {}".format(err)) return -1 class Greeter(BridgeObject): # LightDM.Greeter Signals authentication_complete = Bridge.signal() autologin_timer_expired = Bridge.signal() idle = Bridge.signal() reset = Bridge.signal() show_message = Bridge.signal(str, LightDM.MessageType, arguments=('text', 'type')) show_prompt = Bridge.signal(str, LightDM.PromptType, arguments=('text', 'type')) brightness_update = Bridge.signal() battery_update = Bridge.signal() noop_signal = Bridge.signal() property_changed = Bridge.signal() _battery = None def __init__(self, *args, **kwargs): super().__init__(name='LightDMGreeter', *args, **kwargs) self._config = web_greeter_config["config"] self._shared_data_directory = '' self._themes_directory = web_greeter_config["app"]["theme_dir"] if self._config["features"]["battery"]: self._battery = Battery() try: LightDMGreeter.connect_to_daemon_sync() except Exception as err: logger.error(err) dia = Dialog(title="An error ocurred", message="Detected a problem that could interfere with the system login process", detail="LightDM: {0}\nYou can continue without major problems, but you won't be able to log in".format(err), buttons=["Okay"]) dia.exec() pass self._connect_signals() self._determine_shared_data_directory_path() logger.debug("LightDM API connected") def _determine_shared_data_directory_path(self): user = LightDMUsers.get_users()[0] user_data_dir = LightDMGreeter.ensure_shared_data_dir_sync(user.get_name()) if user_data_dir == None: return self._shared_data_directory = user_data_dir.rpartition('/')[0] def _connect_signals(self): LightDMGreeter.connect( 'authentication-complete', lambda greeter: self._emit_signal(self.authentication_complete) ) LightDMGreeter.connect( 'autologin-timer-expired', lambda greeter: self._emit_signal(self.autologin_timer_expired) ) LightDMGreeter.connect('idle', lambda greeter: self._emit_signal(self.idle)) LightDMGreeter.connect('reset', lambda greeter: self._emit_signal(self.reset)) LightDMGreeter.connect( 'show-message', lambda greeter, msg, mtype: self._emit_signal(self.show_message, msg, mtype) ) LightDMGreeter.connect( 'show-prompt', lambda greeter, msg, mtype: self._emit_signal(self.show_prompt, msg, mtype) ) if self._battery: self._battery.connect(lambda: self.battery_update.emit()) def _emit_signal(self, _signal, *args): self.property_changed.emit() QTimer().singleShot(300, lambda: _signal.emit(*args)) @Bridge.prop(str, notify=property_changed) def authentication_user(self): return LightDMGreeter.get_authentication_user() or '' @Bridge.prop(bool, notify=noop_signal) def autologin_guest(self): return LightDMGreeter.get_autologin_guest_hint() @Bridge.prop(int, notify=noop_signal) def autologin_timeout(self): return LightDMGreeter.get_autologin_timeout_hint() @Bridge.prop(str, notify=noop_signal) def autologin_user(self): return LightDMGreeter.get_autologin_user_hint() @Bridge.prop(QVariant, notify=battery_update) def batteryData(self): return battery_to_dict(self._battery) @Bridge.prop(int, notify=brightness_update) def brightness(self): return getBrightness(self) @brightness.setter def brightness(self, quantity): setBrightness(quantity) @Bridge.prop(bool, notify=noop_signal) def can_hibernate(self): return LightDM.get_can_hibernate() @Bridge.prop(bool, notify=noop_signal) def can_restart(self): return LightDM.get_can_restart() @Bridge.prop(bool, notify=noop_signal) def can_shutdown(self): return LightDM.get_can_shutdown() @Bridge.prop(bool, notify=noop_signal) def can_suspend(self): return LightDM.get_can_suspend() @Bridge.prop(bool, notify=noop_signal) def can_access_brightness(self): return self._config["features"]["backlight"]["enabled"] @Bridge.prop(bool, notify=noop_signal) def can_access_battery(self): return self._config["features"]["battery"] @Bridge.prop(str, notify=noop_signal) def default_session(self): return LightDMGreeter.get_default_session_hint() @Bridge.prop(bool, notify=noop_signal) def has_guest_account(self): return LightDMGreeter.get_has_guest_account_hint() @Bridge.prop(bool, notify=noop_signal) def hide_users_hint(self): return LightDMGreeter.get_hide_users_hint() @Bridge.prop(str, notify=noop_signal) def hostname(self): return LightDM.get_hostname() @Bridge.prop(bool, notify=property_changed) def in_authentication(self): return LightDMGreeter.get_in_authentication() @Bridge.prop(bool, notify=property_changed) def is_authenticated(self): return LightDMGreeter.get_is_authenticated() @Bridge.prop(QVariant, notify=property_changed) def language(self): return language_to_dict(LightDM.get_language()) @Bridge.prop(QVariant, notify=noop_signal) def languages(self): return [language_to_dict(lang) for lang in LightDM.get_languages()] @Bridge.prop(QVariant, notify=noop_signal) def layout(self): return layout_to_dict(LightDM.get_layout()) @layout.setter def layout(self, layout): if type(layout) != dict: return False lay = dict( name = layout.get("name") or "", description = layout.get("description") or "", short_description = layout.get("short_description") or "" ) return LightDM.set_layout(LightDM.Layout(**lay)) @Bridge.prop(QVariant, notify=noop_signal) def layouts(self): return [layout_to_dict(layout) for layout in LightDM.get_layouts()] @Bridge.prop(bool, notify=noop_signal) def lock_hint(self): return LightDMGreeter.get_lock_hint() @Bridge.prop(QVariant, notify=property_changed) def remote_sessions(self): return [session_to_dict(session) for session in LightDM.get_remote_sessions()] @Bridge.prop(bool, notify=noop_signal) def select_guest_hint(self): return LightDMGreeter.get_select_guest_hint() @Bridge.prop(str, notify=noop_signal) def select_user_hint(self): return LightDMGreeter.get_select_user_hint() or '' @Bridge.prop(QVariant, notify=noop_signal) def sessions(self): return [session_to_dict(session) for session in LightDM.get_sessions()] @Bridge.prop(str, notify=noop_signal) def shared_data_directory(self): return self._shared_data_directory or '' @Bridge.prop(bool, notify=noop_signal) def show_manual_login_hint(self): return LightDMGreeter.get_show_manual_login_hint() @Bridge.prop(bool, notify=noop_signal) def show_remote_login_hint(self): return LightDMGreeter.get_show_remote_login_hint() @Bridge.prop(str, notify=noop_signal) def themes_directory(self): return self._themes_directory @Bridge.prop(QVariant, notify=noop_signal) def users(self): return [user_to_dict(user) for user in LightDMUsers.get_users()] @Bridge.method(str) def authenticate(self, username): LightDMGreeter.authenticate(username) self.property_changed.emit() @Bridge.method() def authenticate_as_guest(self): LightDMGreeter.authenticate_as_guest() self.property_changed.emit() @Bridge.method(int) def brightnessSet(self, quantity): setBrightness(quantity) @Bridge.method(int) def brightnessIncrease(self, quantity): increaseBrightness(quantity) @Bridge.method(int) def brightnessDecrease(self, quantity): decreaseBrightness(quantity) @Bridge.method() def cancel_authentication(self): LightDMGreeter.cancel_authentication() self.property_changed.emit() @Bridge.method() def cancel_autologin(self): LightDMGreeter.cancel_autologin() self.property_changed.emit() @Bridge.method(result=bool) def hibernate(self): return LightDM.hibernate() @Bridge.method(str) def respond(self, response): LightDMGreeter.respond(response) self.property_changed.emit() @Bridge.method(result=bool) def restart(self): return LightDM.restart() @Bridge.method(str) def set_language(self, lang): if self.is_authenticated: LightDMGreeter.set_language(lang) self.property_changed.emit() @Bridge.method(result=bool) def shutdown(self): return LightDM.shutdown() @Bridge.method(str, result=bool) def start_session(self, session): if not session.strip(): return started = LightDMGreeter.start_session_sync(session) if started or self.is_authenticated(): reset_screensaver() return started @Bridge.method(result=bool) def suspend(self): return LightDM.suspend()