2022-02-19 12:14:37 +09:00
|
|
|
import asyncio
|
2022-02-23 07:51:29 +09:00
|
|
|
import itertools
|
2022-02-19 12:14:37 +09:00
|
|
|
from functools import wraps
|
|
|
|
|
|
|
|
from quart import current_app
|
|
|
|
|
2022-02-20 16:20:43 +09:00
|
|
|
from anonstream.broadcast import broadcast, broadcast_users_update
|
2022-02-23 07:51:29 +09:00
|
|
|
from anonstream.stream import is_online, get_stream_title, get_stream_uptime
|
2022-02-19 12:14:37 +09:00
|
|
|
from anonstream.wrappers import with_timestamp
|
|
|
|
from anonstream.helpers.user import is_visible
|
|
|
|
|
|
|
|
CONFIG = current_app.config
|
|
|
|
MESSAGES = current_app.messages
|
|
|
|
USERS_BY_TOKEN = current_app.users_by_token
|
|
|
|
USERS = current_app.users
|
2022-02-20 18:09:35 +09:00
|
|
|
CAPTCHAS = current_app.captchas
|
|
|
|
CAPTCHA_SIGNER = current_app.captcha_signer
|
2022-02-19 12:14:37 +09:00
|
|
|
|
2022-02-20 10:06:13 +09:00
|
|
|
async def sleep_and_collect_task(delay):
|
|
|
|
coro = asyncio.sleep(delay)
|
|
|
|
task = asyncio.create_task(coro)
|
|
|
|
current_app.background_sleep.add(task)
|
|
|
|
try:
|
|
|
|
await task
|
|
|
|
finally:
|
|
|
|
current_app.background_sleep.remove(task)
|
|
|
|
|
2022-02-19 12:14:37 +09:00
|
|
|
def with_period(period):
|
|
|
|
def periodically(f):
|
|
|
|
@wraps(f)
|
|
|
|
async def wrapper(*args, **kwargs):
|
2022-02-23 07:51:29 +09:00
|
|
|
for iteration in itertools.count():
|
|
|
|
await f(iteration, *args, **kwargs)
|
2022-02-20 10:06:13 +09:00
|
|
|
try:
|
|
|
|
await sleep_and_collect_task(period)
|
|
|
|
except asyncio.CancelledError:
|
2022-02-19 12:14:37 +09:00
|
|
|
break
|
|
|
|
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
return periodically
|
|
|
|
|
2022-02-20 16:20:43 +09:00
|
|
|
@with_period(CONFIG['TASK_PERIOD_ROTATE_USERS'])
|
2022-02-19 12:14:37 +09:00
|
|
|
@with_timestamp
|
2022-02-23 07:51:29 +09:00
|
|
|
async def t_sunset_users(iteration, timestamp):
|
|
|
|
if iteration == 0:
|
|
|
|
return
|
|
|
|
|
2022-02-19 12:14:37 +09:00
|
|
|
tokens = []
|
|
|
|
for token in USERS_BY_TOKEN:
|
|
|
|
user = USERS_BY_TOKEN[token]
|
|
|
|
if not is_visible(timestamp, MESSAGES, user):
|
|
|
|
tokens.append(token)
|
|
|
|
|
|
|
|
token_hashes = []
|
|
|
|
while tokens:
|
|
|
|
token = tokens.pop()
|
|
|
|
token_hash = USERS_BY_TOKEN.pop(token)['token_hash']
|
|
|
|
token_hashes.append(token_hash)
|
|
|
|
|
2022-02-20 16:20:43 +09:00
|
|
|
# Broadcast a users update, in case any users being
|
|
|
|
# removed have been mutated or are new.
|
|
|
|
broadcast_users_update()
|
|
|
|
|
2022-02-19 12:14:37 +09:00
|
|
|
if token_hashes:
|
|
|
|
broadcast(
|
|
|
|
users=USERS,
|
|
|
|
payload={
|
|
|
|
'type': 'rem-users',
|
|
|
|
'token_hashes': token_hashes,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
2022-02-20 18:09:35 +09:00
|
|
|
@with_period(CONFIG['TASK_PERIOD_ROTATE_CAPTCHAS'])
|
2022-02-23 07:51:29 +09:00
|
|
|
async def t_expire_captchas(iteration):
|
|
|
|
if iteration == 0:
|
|
|
|
return
|
|
|
|
|
2022-02-20 18:09:35 +09:00
|
|
|
to_delete = []
|
|
|
|
for digest in CAPTCHAS:
|
|
|
|
valid = CAPTCHA_SIGNER.validate(
|
|
|
|
digest,
|
|
|
|
max_age=CONFIG['CAPTCHA_LIFETIME'],
|
|
|
|
)
|
|
|
|
if not valid:
|
|
|
|
to_delete.append(digest)
|
2022-02-23 07:51:29 +09:00
|
|
|
|
2022-02-20 18:09:35 +09:00
|
|
|
for digest in to_delete:
|
|
|
|
CAPTCHAS.pop(digest)
|
|
|
|
|
2022-02-20 16:20:43 +09:00
|
|
|
@with_period(CONFIG['TASK_PERIOD_BROADCAST_USERS_UPDATE'])
|
2022-02-23 07:51:29 +09:00
|
|
|
async def t_broadcast_users_update(iteration):
|
|
|
|
if iteration == 0:
|
|
|
|
return
|
|
|
|
else:
|
|
|
|
broadcast_users_update()
|
|
|
|
|
|
|
|
@with_period(CONFIG['TASK_PERIOD_BROADCAST_STREAM_INFO_UPDATE'])
|
|
|
|
async def t_broadcast_stream_info_update(iteration):
|
|
|
|
if iteration == 0:
|
|
|
|
current_app.stream_title = await get_stream_title()
|
|
|
|
current_app.stream_uptime = get_stream_uptime()
|
|
|
|
else:
|
|
|
|
payload = {}
|
|
|
|
|
|
|
|
# Check if the stream title has changed
|
|
|
|
title = await get_stream_title()
|
|
|
|
if current_app.stream_title != title:
|
|
|
|
current_app.stream_title = title
|
|
|
|
payload['title'] = title
|
|
|
|
|
|
|
|
# Check if the stream uptime has changed differently than expected
|
|
|
|
if current_app.stream_uptime is None:
|
|
|
|
expected_uptime = None
|
|
|
|
else:
|
|
|
|
expected_uptime = (
|
|
|
|
current_app.stream_uptime
|
|
|
|
+ CONFIG['TASK_PERIOD_BROADCAST_STREAM_INFO_UPDATE']
|
|
|
|
)
|
|
|
|
uptime = get_stream_uptime()
|
|
|
|
current_app.stream_uptime = uptime
|
|
|
|
if uptime is None and expected_uptime is None:
|
|
|
|
pass
|
|
|
|
elif uptime is None or expected_uptime is None:
|
|
|
|
payload['uptime'] = uptime
|
|
|
|
elif abs(uptime - expected_uptime) >= 0.0625:
|
|
|
|
payload['uptime'] = uptime
|
|
|
|
|
|
|
|
if payload:
|
|
|
|
broadcast(USERS, payload={'type': 'info', **payload})
|
2022-02-20 16:20:43 +09:00
|
|
|
|
2022-02-19 12:14:37 +09:00
|
|
|
current_app.add_background_task(t_sunset_users)
|
2022-02-20 18:09:35 +09:00
|
|
|
current_app.add_background_task(t_expire_captchas)
|
2022-02-20 16:20:43 +09:00
|
|
|
current_app.add_background_task(t_broadcast_users_update)
|
2022-02-23 07:51:29 +09:00
|
|
|
current_app.add_background_task(t_broadcast_stream_info_update)
|