2023-11-22 12:47:34 +01:00
|
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
TODO
|
|
|
|
|
|
|
|
|
|
1) Deprecate "add" (see above) and make it interactive.
|
|
|
|
|
Slixfeed: Do you still want to add this URL to subscription list?
|
|
|
|
|
See: case _ if message_lowercase.startswith("add"):
|
|
|
|
|
|
|
|
|
|
3) Assure message delivery before calling a new task.
|
|
|
|
|
See https://slixmpp.readthedocs.io/en/latest/event_index.html#term-marker_acknowledged
|
|
|
|
|
|
|
|
|
|
4) Do not send updates when busy or away.
|
|
|
|
|
See https://slixmpp.readthedocs.io/en/latest/event_index.html#term-changed_status
|
|
|
|
|
|
2023-11-26 06:48:09 +01:00
|
|
|
|
5) Animate "You have X news items"
|
|
|
|
|
📬️ when sent
|
|
|
|
|
📫️ after sent
|
|
|
|
|
|
2023-11-22 12:47:34 +01:00
|
|
|
|
NOTE
|
|
|
|
|
|
|
|
|
|
1) Self presence
|
|
|
|
|
Apparently, it is possible to view self presence.
|
|
|
|
|
This means that there is no need to store presences in order to switch or restore presence.
|
|
|
|
|
check_readiness
|
|
|
|
|
<presence from="slixfeed@canchat.org/xAPgJLHtMMHF" xml:lang="en" id="ab35c07b63a444d0a7c0a9a0b272f301" to="slixfeed@canchat.org/xAPgJLHtMMHF"><status>📂 Send a URL from a blog or a news website.</status><x xmlns="vcard-temp:x:update"><photo /></x></presence>
|
|
|
|
|
JID: self.boundjid.bare
|
2024-01-24 19:11:39 +01:00
|
|
|
|
MUC: self.alias
|
2023-11-22 12:47:34 +01:00
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
2024-02-11 22:31:31 +01:00
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
TIMEOUT
|
|
|
|
|
|
|
|
|
|
import signal
|
|
|
|
|
|
|
|
|
|
def handler(signum, frame):
|
|
|
|
|
print("Timeout!")
|
|
|
|
|
raise Exception("end of time")
|
|
|
|
|
|
|
|
|
|
# This line will set the alarm for 5 seconds
|
|
|
|
|
|
|
|
|
|
signal.signal(signal.SIGALRM, handler)
|
|
|
|
|
signal.alarm(5)
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
|
# Your command here
|
|
|
|
|
pass
|
|
|
|
|
except Exception as exc:
|
|
|
|
|
print(exc)
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
2023-11-22 12:47:34 +01:00
|
|
|
|
import asyncio
|
|
|
|
|
import logging
|
|
|
|
|
import os
|
2024-01-06 23:03:08 +01:00
|
|
|
|
import slixfeed.action as action
|
2024-02-04 19:56:19 +01:00
|
|
|
|
import slixfeed.config as config
|
2024-03-07 15:52:51 +01:00
|
|
|
|
from slixfeed.config import Config
|
2024-01-18 21:57:49 +01:00
|
|
|
|
# from slixfeed.dt import current_time
|
2024-02-04 19:56:19 +01:00
|
|
|
|
import slixfeed.sqlite as sqlite
|
2023-12-28 15:50:23 +01:00
|
|
|
|
# from xmpp import Slixfeed
|
2024-02-07 01:26:42 +01:00
|
|
|
|
from slixfeed.xmpp.presence import XmppPresence
|
|
|
|
|
from slixfeed.xmpp.message import XmppMessage
|
|
|
|
|
from slixfeed.xmpp.connect import XmppConnect
|
|
|
|
|
from slixfeed.xmpp.utility import get_chat_type
|
2024-01-26 12:34:07 +01:00
|
|
|
|
import time
|
2023-11-22 12:47:34 +01:00
|
|
|
|
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# main_task = []
|
|
|
|
|
# jid_tasker = {}
|
|
|
|
|
# task_manager = {}
|
2023-11-22 12:47:34 +01:00
|
|
|
|
loop = asyncio.get_event_loop()
|
|
|
|
|
|
|
|
|
|
|
2024-02-04 18:08:12 +01:00
|
|
|
|
# def init_tasks(self):
|
|
|
|
|
# global task_ping
|
|
|
|
|
# # if task_ping is None or task_ping.done():
|
|
|
|
|
# # task_ping = asyncio.create_task(ping(self, jid=None))
|
|
|
|
|
# try:
|
|
|
|
|
# task_ping.cancel()
|
|
|
|
|
# except:
|
|
|
|
|
# logging.info('No ping task to cancel')
|
|
|
|
|
# task_ping = asyncio.create_task(ping(self, jid=None))
|
|
|
|
|
|
|
|
|
|
|
2024-02-10 18:53:53 +01:00
|
|
|
|
def task_ping(self):
|
|
|
|
|
# global task_ping_instance
|
2024-02-04 18:08:12 +01:00
|
|
|
|
try:
|
2024-02-10 18:53:53 +01:00
|
|
|
|
self.task_ping_instance.cancel()
|
2024-02-04 18:08:12 +01:00
|
|
|
|
except:
|
|
|
|
|
logging.info('No ping task to cancel.')
|
2024-02-10 18:53:53 +01:00
|
|
|
|
self.task_ping_instance = asyncio.create_task(XmppConnect.ping(self))
|
2024-02-04 18:08:12 +01:00
|
|
|
|
|
|
|
|
|
|
2023-11-22 12:47:34 +01:00
|
|
|
|
"""
|
|
|
|
|
FIXME
|
|
|
|
|
|
|
|
|
|
Tasks don't begin at the same time.
|
|
|
|
|
|
|
|
|
|
This is noticeable when calling "check" before "status".
|
|
|
|
|
|
|
|
|
|
await taskhandler.start_tasks(
|
|
|
|
|
self,
|
|
|
|
|
jid,
|
|
|
|
|
["check", "status"]
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
"""
|
2024-03-07 15:52:51 +01:00
|
|
|
|
async def start_tasks_xmpp(self, jid_bare, tasks=None):
|
2024-02-07 23:24:59 +01:00
|
|
|
|
"""
|
|
|
|
|
NOTE
|
|
|
|
|
|
|
|
|
|
For proper activation of tasks involving task 'interval', it is essential
|
|
|
|
|
to place task 'interval' as the last to start due to await asyncio.sleep()
|
|
|
|
|
which otherwise would postpone tasks that would be set after task 'interval'
|
|
|
|
|
"""
|
2024-03-07 15:52:51 +01:00
|
|
|
|
if jid_bare == self.boundjid.bare:
|
2024-02-04 18:08:12 +01:00
|
|
|
|
return
|
|
|
|
|
try:
|
2024-03-07 15:52:51 +01:00
|
|
|
|
self.task_manager[jid_bare]
|
2024-02-04 18:08:12 +01:00
|
|
|
|
except KeyError as e:
|
2024-03-07 15:52:51 +01:00
|
|
|
|
self.task_manager[jid_bare] = {}
|
2024-02-04 19:19:56 +01:00
|
|
|
|
logging.debug('KeyError:', str(e))
|
2024-03-07 15:52:51 +01:00
|
|
|
|
logging.info('Creating new task manager for JID {}'.format(jid_bare))
|
2024-02-04 18:08:12 +01:00
|
|
|
|
if not tasks:
|
2024-02-07 01:26:42 +01:00
|
|
|
|
tasks = ['status', 'check', 'interval']
|
2024-03-07 15:52:51 +01:00
|
|
|
|
logging.info('Stopping tasks {} for JID {}'.format(tasks, jid_bare))
|
2024-02-04 18:08:12 +01:00
|
|
|
|
for task in tasks:
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# if self.task_manager[jid][task]:
|
2024-02-04 18:08:12 +01:00
|
|
|
|
try:
|
2024-03-07 15:52:51 +01:00
|
|
|
|
self.task_manager[jid_bare][task].cancel()
|
2024-02-04 18:08:12 +01:00
|
|
|
|
except:
|
2024-02-04 19:19:56 +01:00
|
|
|
|
logging.info('No task {} for JID {} (start_tasks_xmpp)'
|
2024-03-07 15:52:51 +01:00
|
|
|
|
.format(task, jid_bare))
|
|
|
|
|
logging.info('Starting tasks {} for JID {}'.format(tasks, jid_bare))
|
2023-11-22 12:47:34 +01:00
|
|
|
|
for task in tasks:
|
2023-11-26 06:48:09 +01:00
|
|
|
|
# print("task:", task)
|
|
|
|
|
# print("tasks:")
|
|
|
|
|
# print(tasks)
|
2023-11-23 17:55:36 +01:00
|
|
|
|
# breakpoint()
|
2023-11-22 12:47:34 +01:00
|
|
|
|
match task:
|
2024-02-04 18:08:12 +01:00
|
|
|
|
case 'check':
|
2024-03-07 15:52:51 +01:00
|
|
|
|
self.task_manager[jid_bare]['check'] = asyncio.create_task(
|
|
|
|
|
check_updates(self, jid_bare))
|
2024-02-04 19:56:19 +01:00
|
|
|
|
case 'status':
|
2024-03-07 15:52:51 +01:00
|
|
|
|
self.task_manager[jid_bare]['status'] = asyncio.create_task(
|
|
|
|
|
task_status(self, jid_bare))
|
2024-02-04 18:08:12 +01:00
|
|
|
|
case 'interval':
|
2024-03-07 15:52:51 +01:00
|
|
|
|
self.task_manager[jid_bare]['interval'] = asyncio.create_task(
|
|
|
|
|
task_send(self, jid_bare))
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# for task in self.task_manager[jid].values():
|
2023-11-23 17:55:36 +01:00
|
|
|
|
# print("task_manager[jid].values()")
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# print(self.task_manager[jid].values())
|
2023-11-23 17:55:36 +01:00
|
|
|
|
# print("task")
|
|
|
|
|
# print(task)
|
|
|
|
|
# print("jid")
|
|
|
|
|
# print(jid)
|
|
|
|
|
# breakpoint()
|
|
|
|
|
# await task
|
|
|
|
|
|
2024-01-02 12:42:41 +01:00
|
|
|
|
|
2024-02-11 22:31:31 +01:00
|
|
|
|
async def task_status(self, jid):
|
|
|
|
|
await action.xmpp_send_status(self, jid)
|
2024-03-08 10:14:36 +01:00
|
|
|
|
refresh_task(self, jid, task_status, 'status', '90')
|
2024-02-11 22:31:31 +01:00
|
|
|
|
|
|
|
|
|
|
2024-03-07 15:52:51 +01:00
|
|
|
|
async def task_send(self, jid_bare):
|
|
|
|
|
jid_file = jid_bare.replace('/', '_')
|
2024-02-10 18:53:53 +01:00
|
|
|
|
db_file = config.get_pathname_to_database(jid_file)
|
2024-03-07 15:52:51 +01:00
|
|
|
|
if jid_bare not in self.settings:
|
|
|
|
|
Config.add_settings_jid(self.settings, jid_bare, db_file)
|
|
|
|
|
update_interval = self.settings[jid_bare]['interval'] or self.settings['default']['interval']
|
2024-02-10 18:53:53 +01:00
|
|
|
|
update_interval = 60 * int(update_interval)
|
2024-03-08 10:14:36 +01:00
|
|
|
|
last_update_time = sqlite.get_last_update_time(db_file)
|
2024-02-10 18:53:53 +01:00
|
|
|
|
if last_update_time:
|
|
|
|
|
last_update_time = float(last_update_time)
|
|
|
|
|
diff = time.time() - last_update_time
|
|
|
|
|
if diff < update_interval:
|
|
|
|
|
next_update_time = update_interval - diff
|
|
|
|
|
await asyncio.sleep(next_update_time) # FIXME!
|
|
|
|
|
|
|
|
|
|
# print("jid :", jid, "\n"
|
|
|
|
|
# "time :", time.time(), "\n"
|
|
|
|
|
# "last_update_time :", last_update_time, "\n"
|
|
|
|
|
# "difference :", diff, "\n"
|
|
|
|
|
# "update interval :", update_interval, "\n"
|
|
|
|
|
# "next_update_time :", next_update_time, "\n"
|
|
|
|
|
# )
|
|
|
|
|
|
|
|
|
|
# elif diff > val:
|
|
|
|
|
# next_update_time = val
|
|
|
|
|
await sqlite.update_last_update_time(db_file)
|
|
|
|
|
else:
|
|
|
|
|
await sqlite.set_last_update_time(db_file)
|
2024-03-07 15:52:51 +01:00
|
|
|
|
await action.xmpp_send_update(self, jid_bare)
|
2024-03-08 10:14:36 +01:00
|
|
|
|
refresh_task(self, jid_bare, task_send, 'interval')
|
2024-03-07 15:52:51 +01:00
|
|
|
|
await start_tasks_xmpp(self, jid_bare, ['status'])
|
2023-11-22 12:47:34 +01:00
|
|
|
|
|
|
|
|
|
|
2024-02-10 18:53:53 +01:00
|
|
|
|
def clean_tasks_xmpp(self, jid, tasks=None):
|
|
|
|
|
if not tasks:
|
|
|
|
|
tasks = ['interval', 'status', 'check']
|
|
|
|
|
logging.info('Stopping tasks {} for JID {}'.format(tasks, jid))
|
|
|
|
|
for task in tasks:
|
|
|
|
|
# if self.task_manager[jid][task]:
|
|
|
|
|
try:
|
|
|
|
|
self.task_manager[jid][task].cancel()
|
|
|
|
|
except:
|
|
|
|
|
logging.debug('No task {} for JID {} (clean_tasks_xmpp)'
|
|
|
|
|
.format(task, jid))
|
|
|
|
|
|
|
|
|
|
|
2024-03-08 10:14:36 +01:00
|
|
|
|
def refresh_task(self, jid_bare, callback, key, val=None):
|
2023-11-22 12:47:34 +01:00
|
|
|
|
"""
|
|
|
|
|
Apply new setting at runtime.
|
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
jid : str
|
|
|
|
|
Jabber ID.
|
|
|
|
|
key : str
|
|
|
|
|
Key.
|
|
|
|
|
val : str, optional
|
|
|
|
|
Value. The default is None.
|
|
|
|
|
"""
|
2024-03-07 15:52:51 +01:00
|
|
|
|
logging.info('Refreshing task {} for JID {}'.format(callback, jid_bare))
|
2023-11-22 12:47:34 +01:00
|
|
|
|
if not val:
|
2024-03-07 15:52:51 +01:00
|
|
|
|
jid_file = jid_bare.replace('/', '_')
|
2024-02-04 19:56:19 +01:00
|
|
|
|
db_file = config.get_pathname_to_database(jid_file)
|
2024-03-07 15:52:51 +01:00
|
|
|
|
if jid_bare not in self.settings:
|
|
|
|
|
Config.add_settings_jid(self.settings, jid_bare, db_file)
|
|
|
|
|
val = self.settings[jid_bare][key] or self.settings['default'][key]
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# if self.task_manager[jid][key]:
|
2024-03-07 15:52:51 +01:00
|
|
|
|
if jid_bare in self.task_manager:
|
2023-11-22 12:47:34 +01:00
|
|
|
|
try:
|
2024-03-07 15:52:51 +01:00
|
|
|
|
self.task_manager[jid_bare][key].cancel()
|
2023-11-22 12:47:34 +01:00
|
|
|
|
except:
|
2024-02-04 18:08:12 +01:00
|
|
|
|
logging.info('No task of type {} to cancel for '
|
2024-03-07 15:52:51 +01:00
|
|
|
|
'JID {} (refresh_task)'.format(key, jid_bare))
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# self.task_manager[jid][key] = loop.call_at(
|
2023-12-28 18:58:49 +01:00
|
|
|
|
# loop.time() + 60 * float(val),
|
|
|
|
|
# loop.create_task,
|
|
|
|
|
# (callback(self, jid))
|
|
|
|
|
# # send_update(jid)
|
|
|
|
|
# )
|
2024-03-07 15:52:51 +01:00
|
|
|
|
self.task_manager[jid_bare][key] = loop.create_task(
|
|
|
|
|
wait_and_run(self, callback, jid_bare, val)
|
2023-11-22 12:47:34 +01:00
|
|
|
|
)
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# self.task_manager[jid][key] = loop.call_later(
|
2023-11-22 12:47:34 +01:00
|
|
|
|
# 60 * float(val),
|
|
|
|
|
# loop.create_task,
|
|
|
|
|
# send_update(jid)
|
|
|
|
|
# )
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# self.task_manager[jid][key] = send_update.loop.call_at(
|
2023-11-22 12:47:34 +01:00
|
|
|
|
# send_update.loop.time() + 60 * val,
|
|
|
|
|
# send_update.loop.create_task,
|
|
|
|
|
# send_update(jid)
|
|
|
|
|
# )
|
|
|
|
|
|
|
|
|
|
|
2024-03-07 15:52:51 +01:00
|
|
|
|
async def wait_and_run(self, callback, jid_bare, val):
|
2023-12-28 18:58:49 +01:00
|
|
|
|
await asyncio.sleep(60 * float(val))
|
2024-03-07 15:52:51 +01:00
|
|
|
|
await callback(self, jid_bare)
|
2023-12-28 18:58:49 +01:00
|
|
|
|
|
|
|
|
|
|
2023-11-22 12:47:34 +01:00
|
|
|
|
# TODO Take this function out of
|
|
|
|
|
# <class 'slixmpp.clientxmpp.ClientXMPP'>
|
2024-03-07 07:56:11 +01:00
|
|
|
|
async def check_updates(self, jid):
|
2023-11-22 12:47:34 +01:00
|
|
|
|
"""
|
|
|
|
|
Start calling for update check up.
|
|
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
|
----------
|
|
|
|
|
jid : str
|
|
|
|
|
Jabber ID.
|
|
|
|
|
"""
|
2024-02-04 18:08:12 +01:00
|
|
|
|
logging.info('Scanning for updates for JID {}'.format(jid))
|
2023-11-22 12:47:34 +01:00
|
|
|
|
while True:
|
2024-01-27 18:15:28 +01:00
|
|
|
|
jid_file = jid.replace('/', '_')
|
2024-02-04 19:56:19 +01:00
|
|
|
|
db_file = config.get_pathname_to_database(jid_file)
|
2024-03-08 10:14:36 +01:00
|
|
|
|
urls = sqlite.get_active_feeds_url(db_file)
|
2024-01-06 23:03:08 +01:00
|
|
|
|
for url in urls:
|
2024-03-07 15:52:51 +01:00
|
|
|
|
await action.scan(self, jid, db_file, url)
|
2024-03-07 20:06:31 +01:00
|
|
|
|
val = self.settings['default']['check']
|
2023-12-18 16:29:32 +01:00
|
|
|
|
await asyncio.sleep(60 * float(val))
|
2023-11-22 12:47:34 +01:00
|
|
|
|
# Schedule to call this function again in 90 minutes
|
|
|
|
|
# loop.call_at(
|
|
|
|
|
# loop.time() + 60 * 90,
|
|
|
|
|
# loop.create_task,
|
|
|
|
|
# self.check_updates(jid)
|
|
|
|
|
# )
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
NOTE
|
|
|
|
|
This is an older system, utilizing local storage instead of XMPP presence.
|
2023-12-08 12:32:01 +01:00
|
|
|
|
This function is good for use with protocols that might not have presence.
|
2024-02-07 23:24:59 +01:00
|
|
|
|
ActivityPub, IRC, LXMF, Matrix, Nostr, SMTP, Tox.
|
2023-11-22 12:47:34 +01:00
|
|
|
|
"""
|
|
|
|
|
async def select_file(self):
|
|
|
|
|
"""
|
|
|
|
|
Initiate actions by JID (Jabber ID).
|
|
|
|
|
"""
|
|
|
|
|
while True:
|
2024-02-04 19:56:19 +01:00
|
|
|
|
db_dir = config.get_default_data_directory()
|
2023-11-22 12:47:34 +01:00
|
|
|
|
if not os.path.isdir(db_dir):
|
2024-02-04 18:08:12 +01:00
|
|
|
|
msg = ('Slixfeed does not work without a database.\n'
|
|
|
|
|
'To create a database, follow these steps:\n'
|
|
|
|
|
'Add Slixfeed contact to your roster.\n'
|
|
|
|
|
'Send a feed to the bot by URL:\n'
|
|
|
|
|
'https://reclaimthenet.org/feed/')
|
2023-12-04 15:41:02 +01:00
|
|
|
|
# print(await current_time(), msg)
|
2023-11-22 12:47:34 +01:00
|
|
|
|
print(msg)
|
|
|
|
|
else:
|
|
|
|
|
os.chdir(db_dir)
|
|
|
|
|
files = os.listdir()
|
|
|
|
|
# TODO Use loop (with gather) instead of TaskGroup
|
|
|
|
|
# for file in files:
|
|
|
|
|
# if file.endswith(".db") and not file.endswith(".db-jour.db"):
|
|
|
|
|
# jid = file[:-3]
|
|
|
|
|
# jid_tasker[jid] = asyncio.create_task(self.task_jid(jid))
|
|
|
|
|
# await jid_tasker[jid]
|
|
|
|
|
async with asyncio.TaskGroup() as tg:
|
|
|
|
|
for file in files:
|
2024-02-04 19:56:19 +01:00
|
|
|
|
if (file.endswith('.db') and
|
|
|
|
|
not file.endswith('.db-jour.db')):
|
2023-11-22 12:47:34 +01:00
|
|
|
|
jid = file[:-3]
|
2024-02-07 23:24:59 +01:00
|
|
|
|
main_task.extend([tg.create_task(self.task_jid(jid))])
|
2023-11-22 12:47:34 +01:00
|
|
|
|
# main_task = [tg.create_task(self.task_jid(jid))]
|
2024-02-07 23:24:59 +01:00
|
|
|
|
# self.task_manager.update({jid: tg})
|