[triggers] Refactor Plugin
- Add logging - Add type annotations - Refactor excecuting events
This commit is contained in:
@@ -1,4 +1,5 @@
|
|||||||
# Copyright (C) 2011-2017 Yann Leboulanger <asterix AT lagaule.org>
|
# Copyright (C) 2011-2017 Yann Leboulanger <asterix AT lagaule.org>
|
||||||
|
# Copyright (C) 2022 Philipp Hörist <philipp AT hoerist.com>
|
||||||
#
|
#
|
||||||
# This file is part of Gajim.
|
# This file is part of Gajim.
|
||||||
#
|
#
|
||||||
@@ -17,15 +18,21 @@
|
|||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from typing import Any
|
||||||
from typing import Optional
|
from typing import Optional
|
||||||
|
from typing import Union
|
||||||
|
|
||||||
|
import logging
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from dataclasses import asdict
|
|
||||||
from functools import partial
|
from functools import partial
|
||||||
|
|
||||||
from gajim.common import app
|
from gajim.common import app
|
||||||
from gajim.common import ged
|
from gajim.common import ged
|
||||||
from gajim.common.events import ApplicationEvent
|
from gajim.common.const import PROPAGATE_EVENT
|
||||||
|
from gajim.common.const import STOP_EVENT
|
||||||
|
from gajim.common.events import Notification
|
||||||
|
from gajim.common.events import GcMessageReceived
|
||||||
|
from gajim.common.events import MessageReceived
|
||||||
from gajim.common.events import Notification
|
from gajim.common.events import Notification
|
||||||
from gajim.common.helpers import exec_command
|
from gajim.common.helpers import exec_command
|
||||||
from gajim.common.helpers import play_sound_file
|
from gajim.common.helpers import play_sound_file
|
||||||
@@ -33,21 +40,14 @@ from gajim.common.helpers import play_sound_file
|
|||||||
from gajim.plugins import GajimPlugin
|
from gajim.plugins import GajimPlugin
|
||||||
from gajim.plugins.plugins_i18n import _
|
from gajim.plugins.plugins_i18n import _
|
||||||
|
|
||||||
|
from triggers.util import log_result
|
||||||
|
from triggers.util import RuleResult
|
||||||
from triggers.gtk.config import ConfigDialog
|
from triggers.gtk.config import ConfigDialog
|
||||||
|
|
||||||
|
log = logging.getLogger('gajim.p.triggers')
|
||||||
|
|
||||||
@dataclass
|
MessageEventsT = Union[GcMessageReceived, MessageReceived]
|
||||||
class ExtendedEvent(Notification):
|
RuleT = dict[str, Any]
|
||||||
origin: Optional[ApplicationEvent] = None
|
|
||||||
show_notification: bool = True
|
|
||||||
command: Optional[str] = None
|
|
||||||
sound_file: Optional[str] = None
|
|
||||||
|
|
||||||
@classmethod
|
|
||||||
def from_event(cls, event) -> ExtendedEvent:
|
|
||||||
attr_dict = asdict(event)
|
|
||||||
attr_dict.pop('name')
|
|
||||||
return cls(**attr_dict, origin=event)
|
|
||||||
|
|
||||||
|
|
||||||
class Triggers(GajimPlugin):
|
class Triggers(GajimPlugin):
|
||||||
@@ -61,38 +61,24 @@ class Triggers(GajimPlugin):
|
|||||||
'notification': (ged.PREGUI, self._on_notification),
|
'notification': (ged.PREGUI, self._on_notification),
|
||||||
'message-received': (ged.PREGUI2, self._on_message_received),
|
'message-received': (ged.PREGUI2, self._on_message_received),
|
||||||
'gc-message-received': (ged.PREGUI2, self._on_message_received),
|
'gc-message-received': (ged.PREGUI2, self._on_message_received),
|
||||||
'presence-received': (ged.PREGUI, self._on_presence_received),
|
# 'presence-received': (ged.PREGUI, self._on_presence_received),
|
||||||
}
|
}
|
||||||
|
|
||||||
def _excecute(self, event) -> bool:
|
|
||||||
if event.command is not None:
|
|
||||||
# Used by Triggers plugin
|
|
||||||
try:
|
|
||||||
exec_command(event.command, use_shell=True)
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
|
|
||||||
if event.sound_file is not None:
|
|
||||||
play_sound_file(event.sound_file)
|
|
||||||
|
|
||||||
if not event.show_notification:
|
|
||||||
# This aborts the event excecution
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
def _on_notification(self, event: Notification) -> bool:
|
def _on_notification(self, event: Notification) -> bool:
|
||||||
extended_event = ExtendedEvent.from_event(event)
|
log.info('Process %s, %s', event.name, event.type)
|
||||||
self._check_all(extended_event,
|
result = self._check_all(event,
|
||||||
self._check_rule_apply_notification,
|
self._check_rule_apply_notification,
|
||||||
self._apply_rule)
|
self._apply_rule)
|
||||||
return self._excecute(extended_event)
|
log.info('Result: %s', result)
|
||||||
|
return self._excecute_notification_rules(result, event)
|
||||||
|
|
||||||
def _on_message_received(self, event: Notification) -> bool:
|
def _on_message_received(self, event: MessageEventsT) -> bool:
|
||||||
extended_event = ExtendedEvent.from_event(event)
|
log.info('Process %s', event.name)
|
||||||
self._check_all(extended_event,
|
result = self._check_all(event,
|
||||||
self._check_rule_apply_msg_received,
|
self._check_rule_apply_msg_received,
|
||||||
self._apply_rule)
|
self._apply_rule)
|
||||||
return self._excecute(extended_event)
|
log.info('Result: %s', result)
|
||||||
|
return self._excecute_message_rules(result)
|
||||||
|
|
||||||
def _on_presence_received(self, event):
|
def _on_presence_received(self, event):
|
||||||
# TODO
|
# TODO
|
||||||
@@ -106,19 +92,18 @@ class Triggers(GajimPlugin):
|
|||||||
check_func = self._check_rule_apply_status_changed
|
check_func = self._check_rule_apply_status_changed
|
||||||
self._check_all(event, check_func, self._apply_rule)
|
self._check_all(event, check_func, self._apply_rule)
|
||||||
|
|
||||||
def _check_all(self, event, check_func, apply_func):
|
def _check_all(self, event, check_func, apply_func) -> RuleResult:
|
||||||
# check rules in order
|
result = RuleResult()
|
||||||
|
|
||||||
rules_num = [int(item) for item in self.config.keys()]
|
rules_num = [int(item) for item in self.config.keys()]
|
||||||
rules_num.sort()
|
rules_num.sort()
|
||||||
to_remove = []
|
to_remove = []
|
||||||
for num in rules_num:
|
for num in rules_num:
|
||||||
rule = self.config[str(num)]
|
rule = self.config[str(num)]
|
||||||
if check_func(event, rule):
|
if check_func(event, rule):
|
||||||
apply_func(event, rule)
|
apply_func(result, rule)
|
||||||
if 'one_shot' in rule and rule['one_shot']:
|
if 'one_shot' in rule and rule['one_shot']:
|
||||||
to_remove.append(num)
|
to_remove.append(num)
|
||||||
# Should we stop after first valid rule ?
|
|
||||||
# break
|
|
||||||
|
|
||||||
decal = 0
|
decal = 0
|
||||||
num = 0
|
num = 0
|
||||||
@@ -133,24 +118,35 @@ class Triggers(GajimPlugin):
|
|||||||
else:
|
else:
|
||||||
num += 1
|
num += 1
|
||||||
|
|
||||||
def _check_rule_apply_msg_received(self, event, rule):
|
return result
|
||||||
|
|
||||||
|
@log_result
|
||||||
|
def _check_rule_apply_msg_received(self,
|
||||||
|
event: MessageEventsT,
|
||||||
|
rule: RuleT) -> bool:
|
||||||
return self._check_rule_all('message_received', event, rule)
|
return self._check_rule_all('message_received', event, rule)
|
||||||
|
|
||||||
def _check_rule_apply_connected(self, event, rule):
|
@log_result
|
||||||
|
def _check_rule_apply_connected(self, event, rule: RuleT) -> bool:
|
||||||
return self._check_rule_all('contact_connected', event, rule)
|
return self._check_rule_all('contact_connected', event, rule)
|
||||||
|
|
||||||
def _check_rule_apply_disconnected(self, event, rule):
|
@log_result
|
||||||
|
def _check_rule_apply_disconnected(self, event, rule: RuleT) -> bool:
|
||||||
return self._check_rule_all('contact_disconnected', event, rule)
|
return self._check_rule_all('contact_disconnected', event, rule)
|
||||||
|
|
||||||
def _check_rule_apply_status_changed(self, event, rule):
|
@log_result
|
||||||
|
def _check_rule_apply_status_changed(self, event, rule: RuleT) -> bool:
|
||||||
return self._check_rule_all('contact_status_change', event, rule)
|
return self._check_rule_all('contact_status_change', event, rule)
|
||||||
|
|
||||||
def _check_rule_apply_notification(self, event, rule):
|
@log_result
|
||||||
|
def _check_rule_apply_notification(self,
|
||||||
|
event: Notification,
|
||||||
|
rule: RuleT) -> bool:
|
||||||
# Check notification type
|
# Check notification type
|
||||||
notif_type = ''
|
notif_type = ''
|
||||||
if event.notif_type == 'incoming-message':
|
if event.type == 'incoming-message':
|
||||||
notif_type = 'message_received'
|
notif_type = 'message_received'
|
||||||
if event.notif_type == 'pres':
|
if event.type == 'pres':
|
||||||
# TODO:
|
# TODO:
|
||||||
if (event.base_event.old_show < 2 and
|
if (event.base_event.old_show < 2 and
|
||||||
event.base_event.new_show > 1):
|
event.base_event.new_show > 1):
|
||||||
@@ -163,7 +159,7 @@ class Triggers(GajimPlugin):
|
|||||||
|
|
||||||
return self._check_rule_all(notif_type, event, rule)
|
return self._check_rule_all(notif_type, event, rule)
|
||||||
|
|
||||||
def _check_rule_all(self, notif_type, event, rule):
|
def _check_rule_all(self, notif_type: str, event: Any, rule: RuleT) -> bool:
|
||||||
# Check notification type
|
# Check notification type
|
||||||
if rule['event'] != notif_type:
|
if rule['event'] != notif_type:
|
||||||
return False
|
return False
|
||||||
@@ -187,7 +183,8 @@ class Triggers(GajimPlugin):
|
|||||||
# All is ok
|
# All is ok
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def _check_rule_recipients(self, event, rule):
|
@log_result
|
||||||
|
def _check_rule_recipients(self, event, rule: RuleT) -> bool:
|
||||||
rule_recipients = [t.strip() for t in rule['recipients'].split(',')]
|
rule_recipients = [t.strip() for t in rule['recipients'].split(',')]
|
||||||
if rule['recipient_type'] == 'groupchat':
|
if rule['recipient_type'] == 'groupchat':
|
||||||
if event.jid in rule_recipients:
|
if event.jid in rule_recipients:
|
||||||
@@ -209,7 +206,8 @@ class Triggers(GajimPlugin):
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def _check_rule_status(self, event, rule):
|
@log_result
|
||||||
|
def _check_rule_status(self, event, rule: RuleT) -> bool:
|
||||||
rule_statuses = rule['status'].split()
|
rule_statuses = rule['status'].split()
|
||||||
our_status = app.connections[event.account].status
|
our_status = app.connections[event.account].status
|
||||||
if rule['status'] != 'all' and our_status not in rule_statuses:
|
if rule['status'] != 'all' and our_status not in rule_statuses:
|
||||||
@@ -217,7 +215,8 @@ class Triggers(GajimPlugin):
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def _check_rule_tab_opened(self, event, rule):
|
@log_result
|
||||||
|
def _check_rule_tab_opened(self, event, rule: RuleT) -> bool:
|
||||||
if rule['tab_opened'] == 'both':
|
if rule['tab_opened'] == 'both':
|
||||||
return True
|
return True
|
||||||
tab_opened = False
|
tab_opened = False
|
||||||
@@ -230,7 +229,8 @@ class Triggers(GajimPlugin):
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def _check_rule_has_focus(self, event, rule):
|
@log_result
|
||||||
|
def _check_rule_has_focus(self, event, rule: RuleT) -> bool:
|
||||||
if rule['has_focus'] == 'both':
|
if rule['has_focus'] == 'both':
|
||||||
return True
|
return True
|
||||||
if rule['tab_opened'] == 'no':
|
if rule['tab_opened'] == 'no':
|
||||||
@@ -248,19 +248,42 @@ class Triggers(GajimPlugin):
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def _apply_rule(self, event, rule):
|
def _apply_rule(self, result: RuleResult, rule: RuleT) -> None:
|
||||||
if rule['sound'] == 'no':
|
if rule['sound'] == 'no':
|
||||||
event.origin.sound = None
|
result.sound = False
|
||||||
event.sound_file = None
|
result.sound_file = None
|
||||||
|
|
||||||
elif rule['sound'] == 'yes':
|
elif rule['sound'] == 'yes':
|
||||||
event.origin.sound = None
|
result.sound = False
|
||||||
event.sound_file = rule['sound_file']
|
result.sound_file = rule['sound_file']
|
||||||
|
|
||||||
if rule['run_command']:
|
if rule['run_command']:
|
||||||
event.command = rule['command']
|
result.command = rule['command']
|
||||||
|
|
||||||
if rule['popup'] == 'no':
|
if rule['popup'] == 'no':
|
||||||
event.show_notification = False
|
result.show_notification = False
|
||||||
elif rule['popup'] == 'yes':
|
elif rule['popup'] == 'yes':
|
||||||
event.show_notification = True
|
result.show_notification = True
|
||||||
|
|
||||||
|
def _excecute_notification_rules(self,
|
||||||
|
result: RuleResult,
|
||||||
|
event: Notification) -> bool:
|
||||||
|
if result.sound is False:
|
||||||
|
event.sound = None
|
||||||
|
|
||||||
|
if result.sound_file is not None:
|
||||||
|
play_sound_file(result.sound_file)
|
||||||
|
|
||||||
|
if result.show_notification is False:
|
||||||
|
return STOP_EVENT
|
||||||
|
return PROPAGATE_EVENT
|
||||||
|
|
||||||
|
def _excecute_message_rules(self, result: RuleResult) -> bool:
|
||||||
|
|
||||||
|
if result.command is not None:
|
||||||
|
try:
|
||||||
|
exec_command(result.command, use_shell=True)
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return PROPAGATE_EVENT
|
||||||
|
|||||||
36
triggers/util.py
Normal file
36
triggers/util.py
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
# This file is part of Gajim.
|
||||||
|
#
|
||||||
|
# Gajim 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; version 3 only.
|
||||||
|
#
|
||||||
|
# Gajim 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 Gajim. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from typing import Optional
|
||||||
|
|
||||||
|
import logging
|
||||||
|
from dataclasses import dataclass
|
||||||
|
|
||||||
|
log = logging.getLogger('gajim.p.triggers')
|
||||||
|
|
||||||
|
|
||||||
|
def log_result(func):
|
||||||
|
def wrapper(self, event, rule):
|
||||||
|
res = func(self, event, rule)
|
||||||
|
log.info(f'{event.name} -> {func.__name__} -> {res}')
|
||||||
|
return res
|
||||||
|
return wrapper
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class RuleResult:
|
||||||
|
show_notification: Optional[bool] = None
|
||||||
|
command: Optional[str] = None
|
||||||
|
sound: Optional[bool] = None
|
||||||
|
sound_file: Optional[str] = None
|
||||||
Reference in New Issue
Block a user