funkwhale/api/funkwhale_api/federation/tasks.py

150 wiersze
4.7 KiB
Python
Czysty Zwykły widok Historia

import datetime
2018-04-12 17:57:43 +00:00
import logging
import os
2018-04-12 17:57:43 +00:00
from django.conf import settings
from django.db.models import Q, F
2018-04-12 18:38:06 +00:00
from django.utils import timezone
from dynamic_preferences.registries import global_preferences_registry
2018-06-10 08:55:16 +00:00
from requests.exceptions import RequestException
2018-04-12 16:41:43 +00:00
from funkwhale_api.common import preferences
2018-04-12 17:57:43 +00:00
from funkwhale_api.common import session
from funkwhale_api.music import models as music_models
2018-04-11 21:13:33 +00:00
from funkwhale_api.taskapp import celery
2018-06-10 08:55:16 +00:00
from . import models, signing
from . import routes
2018-04-12 17:57:43 +00:00
logger = logging.getLogger(__name__)
2018-06-09 13:36:16 +00:00
@celery.app.task(name="federation.clean_music_cache")
def clean_music_cache():
preferences = global_preferences_registry.manager()
2018-06-09 13:36:16 +00:00
delay = preferences["federation__music_cache_duration"]
if delay < 1:
return # cache clearing disabled
limit = timezone.now() - datetime.timedelta(minutes=delay)
2018-06-09 13:36:16 +00:00
candidates = (
2018-09-22 12:29:30 +00:00
music_models.Upload.objects.filter(
2018-06-09 13:36:16 +00:00
Q(audio_file__isnull=False)
& (Q(accessed_date__lt=limit) | Q(accessed_date=None)),
# library__actor__user=None,
)
.local(False)
2018-06-09 13:36:16 +00:00
.exclude(audio_file="")
.only("audio_file", "id")
.order_by("id")
2018-06-09 13:36:16 +00:00
)
2018-09-22 12:29:30 +00:00
for upload in candidates:
upload.audio_file.delete()
# we also delete orphaned files, if any
2018-06-09 13:36:16 +00:00
storage = models.LibraryTrack._meta.get_field("audio_file").storage
files = get_files(storage, "federation_cache/tracks")
2018-09-22 12:29:30 +00:00
existing = music_models.Upload.objects.filter(audio_file__in=files)
2018-06-09 13:36:16 +00:00
missing = set(files) - set(existing.values_list("audio_file", flat=True))
for m in missing:
storage.delete(m)
def get_files(storage, *parts):
"""
This is a recursive function that return all files available
in a given directory using django's storage.
"""
if not parts:
2018-06-09 13:36:16 +00:00
raise ValueError("Missing path")
try:
dirs, files = storage.listdir(os.path.join(*parts))
except FileNotFoundError:
return []
for dir in dirs:
files += get_files(storage, *(list(parts) + [dir]))
2018-06-09 13:36:16 +00:00
return [os.path.join(parts[-1], path) for path in files]
@celery.app.task(name="federation.dispatch_inbox")
@celery.require_instance(models.Activity.objects.select_related(), "activity")
def dispatch_inbox(activity):
"""
Given an activity instance, triggers our internal delivery logic (follow
creation, etc.)
"""
2018-09-22 12:29:30 +00:00
routes.inbox.dispatch(
activity.payload,
context={
"activity": activity,
"actor": activity.actor,
"inbox_items": activity.inbox_items.filter(is_read=False),
},
)
@celery.app.task(name="federation.dispatch_outbox")
@celery.require_instance(models.Activity.objects.select_related(), "activity")
def dispatch_outbox(activity):
"""
2018-09-22 12:29:30 +00:00
Deliver a local activity to its recipients, both locally and remotely
"""
2018-09-22 12:29:30 +00:00
inbox_items = activity.inbox_items.filter(is_read=False).select_related()
2018-09-22 12:29:30 +00:00
if inbox_items.exists():
dispatch_inbox.delay(activity_id=activity.pk)
if not preferences.get("federation__enabled"):
# federation is disabled, we only deliver to local recipients
return
deliveries = activity.deliveries.filter(is_delivered=False)
2018-09-22 12:29:30 +00:00
for id in deliveries.values_list("pk", flat=True):
deliver_to_remote.delay(delivery_id=id)
@celery.app.task(
name="federation.deliver_to_remote_inbox",
autoretry_for=[RequestException],
retry_backoff=30,
max_retries=5,
)
2018-09-22 12:29:30 +00:00
@celery.require_instance(
models.Delivery.objects.filter(is_delivered=False).select_related(
"activity__actor"
),
"delivery",
)
def deliver_to_remote(delivery):
if not preferences.get("federation__enabled"):
# federation is disabled, we only deliver to local recipients
return
2018-09-22 12:29:30 +00:00
actor = delivery.activity.actor
logger.info("Preparing activity delivery to %s", delivery.inbox_url)
auth = signing.get_auth(actor.private_key, actor.private_key_id)
try:
response = session.get_session().post(
auth=auth,
2018-09-22 12:29:30 +00:00
json=delivery.activity.payload,
url=delivery.inbox_url,
timeout=5,
verify=settings.EXTERNAL_REQUESTS_VERIFY_SSL,
headers={"Content-Type": "application/activity+json"},
)
logger.debug("Remote answered with %s", response.status_code)
response.raise_for_status()
except Exception:
2018-09-22 12:29:30 +00:00
delivery.last_attempt_date = timezone.now()
delivery.attempts = F("attempts") + 1
delivery.save(update_fields=["last_attempt_date", "attempts"])
raise
else:
2018-09-22 12:29:30 +00:00
delivery.last_attempt_date = timezone.now()
delivery.attempts = F("attempts") + 1
delivery.is_delivered = True
delivery.save(update_fields=["last_attempt_date", "attempts", "is_delivered"])