Admin UI for libraries and uploads
This commit is contained in:
parent
9aee135c2f
commit
a605bcbe76
27 changed files with 2140 additions and 361 deletions
|
|
@ -1,4 +1,8 @@
|
|||
from django import forms
|
||||
from django.db.models import Q
|
||||
from django.conf import settings
|
||||
|
||||
import django_filters
|
||||
from django_filters import rest_framework as filters
|
||||
|
||||
from funkwhale_api.common import fields
|
||||
|
|
@ -11,19 +15,32 @@ from funkwhale_api.music import models as music_models
|
|||
from funkwhale_api.users import models as users_models
|
||||
|
||||
|
||||
class ManageUploadFilterSet(filters.FilterSet):
|
||||
q = fields.SearchFilter(
|
||||
search_fields=[
|
||||
"track__title",
|
||||
"track__album__title",
|
||||
"track__artist__name",
|
||||
"source",
|
||||
]
|
||||
)
|
||||
class ActorField(forms.CharField):
|
||||
def clean(self, value):
|
||||
value = super().clean(value)
|
||||
if not value:
|
||||
return value
|
||||
|
||||
class Meta:
|
||||
model = music_models.Upload
|
||||
fields = ["q", "track__album", "track__artist", "track"]
|
||||
parts = value.split("@")
|
||||
|
||||
return {
|
||||
"username": parts[0],
|
||||
"domain": parts[1] if len(parts) > 1 else settings.FEDERATION_HOSTNAME,
|
||||
}
|
||||
|
||||
|
||||
def get_actor_filter(actor_field):
|
||||
def handler(v):
|
||||
if not v:
|
||||
return Q(**{actor_field: None})
|
||||
return Q(
|
||||
**{
|
||||
"{}__preferred_username__iexact".format(actor_field): v["username"],
|
||||
"{}__domain__name__iexact".format(actor_field): v["domain"],
|
||||
}
|
||||
)
|
||||
|
||||
return {"field": ActorField(), "handler": handler}
|
||||
|
||||
|
||||
class ManageArtistFilterSet(filters.FilterSet):
|
||||
|
|
@ -37,7 +54,11 @@ class ManageArtistFilterSet(filters.FilterSet):
|
|||
filter_fields={
|
||||
"domain": {
|
||||
"handler": lambda v: federation_utils.get_domain_query_from_url(v)
|
||||
}
|
||||
},
|
||||
"library_id": {
|
||||
"to": "tracks__uploads__library_id",
|
||||
"field": forms.IntegerField(),
|
||||
},
|
||||
},
|
||||
)
|
||||
)
|
||||
|
|
@ -61,6 +82,10 @@ class ManageAlbumFilterSet(filters.FilterSet):
|
|||
"domain": {
|
||||
"handler": lambda v: federation_utils.get_domain_query_from_url(v)
|
||||
},
|
||||
"library_id": {
|
||||
"to": "tracks__uploads__library_id",
|
||||
"field": forms.IntegerField(),
|
||||
},
|
||||
},
|
||||
)
|
||||
)
|
||||
|
|
@ -93,6 +118,10 @@ class ManageTrackFilterSet(filters.FilterSet):
|
|||
"domain": {
|
||||
"handler": lambda v: federation_utils.get_domain_query_from_url(v)
|
||||
},
|
||||
"library_id": {
|
||||
"to": "uploads__library_id",
|
||||
"field": forms.IntegerField(),
|
||||
},
|
||||
},
|
||||
)
|
||||
)
|
||||
|
|
@ -102,6 +131,96 @@ class ManageTrackFilterSet(filters.FilterSet):
|
|||
fields = ["q", "title", "mbid", "fid", "artist", "album", "license"]
|
||||
|
||||
|
||||
class ManageLibraryFilterSet(filters.FilterSet):
|
||||
ordering = django_filters.OrderingFilter(
|
||||
# tuple-mapping retains order
|
||||
fields=(
|
||||
("creation_date", "creation_date"),
|
||||
("_uploads_count", "uploads_count"),
|
||||
("followers_count", "followers_count"),
|
||||
)
|
||||
)
|
||||
q = fields.SmartSearchFilter(
|
||||
config=search.SearchConfig(
|
||||
search_fields={
|
||||
"name": {"to": "name"},
|
||||
"description": {"to": "description"},
|
||||
"fid": {"to": "fid"},
|
||||
},
|
||||
filter_fields={
|
||||
"artist_id": {
|
||||
"to": "uploads__track__artist_id",
|
||||
"field": forms.IntegerField(),
|
||||
},
|
||||
"album_id": {
|
||||
"to": "uploads__track__album_id",
|
||||
"field": forms.IntegerField(),
|
||||
},
|
||||
"track_id": {"to": "uploads__track__id", "field": forms.IntegerField()},
|
||||
"domain": {"to": "actor__domain_id"},
|
||||
"account": get_actor_filter("actor"),
|
||||
"privacy_level": {"to": "privacy_level"},
|
||||
},
|
||||
)
|
||||
)
|
||||
domain = filters.CharFilter("actor__domain_id")
|
||||
|
||||
class Meta:
|
||||
model = music_models.Library
|
||||
fields = ["q", "name", "fid", "privacy_level", "domain"]
|
||||
|
||||
|
||||
class ManageUploadFilterSet(filters.FilterSet):
|
||||
ordering = django_filters.OrderingFilter(
|
||||
# tuple-mapping retains order
|
||||
fields=(
|
||||
("creation_date", "creation_date"),
|
||||
("modification_date", "modification_date"),
|
||||
("accessed_date", "accessed_date"),
|
||||
("size", "size"),
|
||||
("bitrate", "bitrate"),
|
||||
("duration", "duration"),
|
||||
)
|
||||
)
|
||||
q = fields.SmartSearchFilter(
|
||||
config=search.SearchConfig(
|
||||
search_fields={
|
||||
"source": {"to": "source"},
|
||||
"fid": {"to": "fid"},
|
||||
"track": {"to": "track__title"},
|
||||
"album": {"to": "track__album__title"},
|
||||
"artist": {"to": "track__artist__name"},
|
||||
},
|
||||
filter_fields={
|
||||
"library_id": {"to": "library_id", "field": forms.IntegerField()},
|
||||
"artist_id": {"to": "track__artist_id", "field": forms.IntegerField()},
|
||||
"album_id": {"to": "track__album_id", "field": forms.IntegerField()},
|
||||
"track_id": {"to": "track__id", "field": forms.IntegerField()},
|
||||
"domain": {"to": "library__actor__domain_id"},
|
||||
"import_reference": {"to": "import_reference"},
|
||||
"type": {"to": "mimetype"},
|
||||
"status": {"to": "import_status"},
|
||||
"account": get_actor_filter("library__actor"),
|
||||
"privacy_level": {"to": "library__privacy_level"},
|
||||
},
|
||||
)
|
||||
)
|
||||
domain = filters.CharFilter("library__actor__domain_id")
|
||||
privacy_level = filters.CharFilter("library__privacy_level")
|
||||
|
||||
class Meta:
|
||||
model = music_models.Upload
|
||||
fields = [
|
||||
"q",
|
||||
"fid",
|
||||
"privacy_level",
|
||||
"domain",
|
||||
"mimetype",
|
||||
"import_reference",
|
||||
"import_status",
|
||||
]
|
||||
|
||||
|
||||
class ManageDomainFilterSet(filters.FilterSet):
|
||||
q = fields.SearchFilter(search_fields=["name"])
|
||||
|
||||
|
|
|
|||
|
|
@ -15,67 +15,6 @@ from funkwhale_api.users import models as users_models
|
|||
from . import filters
|
||||
|
||||
|
||||
class ManageUploadArtistSerializer(serializers.ModelSerializer):
|
||||
class Meta:
|
||||
model = music_models.Artist
|
||||
fields = ["id", "mbid", "creation_date", "name"]
|
||||
|
||||
|
||||
class ManageUploadAlbumSerializer(serializers.ModelSerializer):
|
||||
artist = ManageUploadArtistSerializer()
|
||||
|
||||
class Meta:
|
||||
model = music_models.Album
|
||||
fields = (
|
||||
"id",
|
||||
"mbid",
|
||||
"title",
|
||||
"artist",
|
||||
"release_date",
|
||||
"cover",
|
||||
"creation_date",
|
||||
)
|
||||
|
||||
|
||||
class ManageUploadTrackSerializer(serializers.ModelSerializer):
|
||||
artist = ManageUploadArtistSerializer()
|
||||
album = ManageUploadAlbumSerializer()
|
||||
|
||||
class Meta:
|
||||
model = music_models.Track
|
||||
fields = ("id", "mbid", "title", "album", "artist", "creation_date", "position")
|
||||
|
||||
|
||||
class ManageUploadSerializer(serializers.ModelSerializer):
|
||||
track = ManageUploadTrackSerializer()
|
||||
|
||||
class Meta:
|
||||
model = music_models.Upload
|
||||
fields = (
|
||||
"id",
|
||||
"path",
|
||||
"source",
|
||||
"filename",
|
||||
"mimetype",
|
||||
"track",
|
||||
"duration",
|
||||
"mimetype",
|
||||
"creation_date",
|
||||
"bitrate",
|
||||
"size",
|
||||
"path",
|
||||
)
|
||||
|
||||
|
||||
class ManageUploadActionSerializer(common_serializers.ActionSerializer):
|
||||
actions = [common_serializers.Action("delete", allow_all=False)]
|
||||
filterset_class = filters.ManageUploadFilterSet
|
||||
|
||||
@transaction.atomic
|
||||
def handle_delete(self, objects):
|
||||
return objects.delete()
|
||||
|
||||
|
||||
class PermissionsSerializer(serializers.Serializer):
|
||||
def to_representation(self, o):
|
||||
return o.get_permissions(defaults=self.context.get("default_permissions"))
|
||||
|
|
@ -493,3 +432,111 @@ class ManageArtistActionSerializer(common_serializers.ActionSerializer):
|
|||
@transaction.atomic
|
||||
def handle_delete(self, objects):
|
||||
return objects.delete()
|
||||
|
||||
|
||||
class ManageLibraryActionSerializer(common_serializers.ActionSerializer):
|
||||
actions = [common_serializers.Action("delete", allow_all=False)]
|
||||
filterset_class = filters.ManageLibraryFilterSet
|
||||
|
||||
@transaction.atomic
|
||||
def handle_delete(self, objects):
|
||||
return objects.delete()
|
||||
|
||||
|
||||
class ManageUploadActionSerializer(common_serializers.ActionSerializer):
|
||||
actions = [common_serializers.Action("delete", allow_all=False)]
|
||||
filterset_class = filters.ManageUploadFilterSet
|
||||
|
||||
@transaction.atomic
|
||||
def handle_delete(self, objects):
|
||||
return objects.delete()
|
||||
|
||||
|
||||
class ManageLibrarySerializer(serializers.ModelSerializer):
|
||||
domain = serializers.CharField(source="domain_name")
|
||||
actor = ManageBaseActorSerializer()
|
||||
uploads_count = serializers.SerializerMethodField()
|
||||
followers_count = serializers.SerializerMethodField()
|
||||
|
||||
class Meta:
|
||||
model = music_models.Library
|
||||
fields = [
|
||||
"id",
|
||||
"uuid",
|
||||
"fid",
|
||||
"url",
|
||||
"name",
|
||||
"description",
|
||||
"domain",
|
||||
"is_local",
|
||||
"creation_date",
|
||||
"privacy_level",
|
||||
"uploads_count",
|
||||
"followers_count",
|
||||
"followers_url",
|
||||
"actor",
|
||||
]
|
||||
|
||||
def get_uploads_count(self, obj):
|
||||
return getattr(obj, "_uploads_count", obj.uploads_count)
|
||||
|
||||
def get_followers_count(self, obj):
|
||||
return getattr(obj, "followers_count", None)
|
||||
|
||||
|
||||
class ManageNestedLibrarySerializer(serializers.ModelSerializer):
|
||||
domain = serializers.CharField(source="domain_name")
|
||||
actor = ManageBaseActorSerializer()
|
||||
|
||||
class Meta:
|
||||
model = music_models.Library
|
||||
fields = [
|
||||
"id",
|
||||
"uuid",
|
||||
"fid",
|
||||
"url",
|
||||
"name",
|
||||
"description",
|
||||
"domain",
|
||||
"is_local",
|
||||
"creation_date",
|
||||
"privacy_level",
|
||||
"followers_url",
|
||||
"actor",
|
||||
]
|
||||
|
||||
|
||||
class ManageUploadSerializer(serializers.ModelSerializer):
|
||||
track = ManageNestedTrackSerializer()
|
||||
library = ManageNestedLibrarySerializer()
|
||||
domain = serializers.CharField(source="domain_name")
|
||||
|
||||
class Meta:
|
||||
model = music_models.Upload
|
||||
fields = (
|
||||
"id",
|
||||
"uuid",
|
||||
"fid",
|
||||
"domain",
|
||||
"is_local",
|
||||
"audio_file",
|
||||
"listen_url",
|
||||
"source",
|
||||
"filename",
|
||||
"mimetype",
|
||||
"duration",
|
||||
"mimetype",
|
||||
"bitrate",
|
||||
"size",
|
||||
"creation_date",
|
||||
"accessed_date",
|
||||
"modification_date",
|
||||
"metadata",
|
||||
"import_date",
|
||||
"import_details",
|
||||
"import_status",
|
||||
"import_metadata",
|
||||
"import_reference",
|
||||
"track",
|
||||
"library",
|
||||
)
|
||||
|
|
|
|||
|
|
@ -7,10 +7,11 @@ federation_router = routers.SimpleRouter()
|
|||
federation_router.register(r"domains", views.ManageDomainViewSet, "domains")
|
||||
|
||||
library_router = routers.SimpleRouter()
|
||||
library_router.register(r"uploads", views.ManageUploadViewSet, "uploads")
|
||||
library_router.register(r"artists", views.ManageArtistViewSet, "artists")
|
||||
library_router.register(r"albums", views.ManageAlbumViewSet, "albums")
|
||||
library_router.register(r"artists", views.ManageArtistViewSet, "artists")
|
||||
library_router.register(r"libraries", views.ManageLibraryViewSet, "libraries")
|
||||
library_router.register(r"tracks", views.ManageTrackViewSet, "tracks")
|
||||
library_router.register(r"uploads", views.ManageUploadViewSet, "uploads")
|
||||
|
||||
moderation_router = routers.SimpleRouter()
|
||||
moderation_router.register(
|
||||
|
|
|
|||
|
|
@ -19,38 +19,6 @@ from funkwhale_api.users import models as users_models
|
|||
from . import filters, serializers
|
||||
|
||||
|
||||
class ManageUploadViewSet(
|
||||
mixins.ListModelMixin, mixins.RetrieveModelMixin, viewsets.GenericViewSet
|
||||
):
|
||||
queryset = (
|
||||
music_models.Upload.objects.all()
|
||||
.select_related("track__artist", "track__album__artist")
|
||||
.order_by("-id")
|
||||
)
|
||||
serializer_class = serializers.ManageUploadSerializer
|
||||
filterset_class = filters.ManageUploadFilterSet
|
||||
required_scope = "instance:libraries"
|
||||
ordering_fields = [
|
||||
"accessed_date",
|
||||
"modification_date",
|
||||
"creation_date",
|
||||
"track__artist__name",
|
||||
"bitrate",
|
||||
"size",
|
||||
"duration",
|
||||
]
|
||||
|
||||
@rest_decorators.action(methods=["post"], detail=False)
|
||||
def action(self, request, *args, **kwargs):
|
||||
queryset = self.get_queryset()
|
||||
serializer = serializers.ManageUploadActionSerializer(
|
||||
request.data, queryset=queryset
|
||||
)
|
||||
serializer.is_valid(raise_exception=True)
|
||||
result = serializer.save()
|
||||
return response.Response(result, status=200)
|
||||
|
||||
|
||||
def get_stats(tracks, target):
|
||||
data = {}
|
||||
tracks = list(tracks.values_list("pk", flat=True))
|
||||
|
|
@ -70,6 +38,12 @@ def get_stats(tracks, target):
|
|||
).count()
|
||||
data["libraries"] = uploads.values_list("library", flat=True).distinct().count()
|
||||
data["uploads"] = uploads.count()
|
||||
data.update(get_media_stats(uploads))
|
||||
return data
|
||||
|
||||
|
||||
def get_media_stats(uploads):
|
||||
data = {}
|
||||
data["media_total_size"] = uploads.aggregate(v=Sum("size"))["v"] or 0
|
||||
data["media_downloaded_size"] = (
|
||||
uploads.with_file().aggregate(v=Sum("size"))["v"] or 0
|
||||
|
|
@ -85,6 +59,7 @@ class ManageArtistViewSet(
|
|||
):
|
||||
queryset = (
|
||||
music_models.Artist.objects.all()
|
||||
.distinct()
|
||||
.order_by("-id")
|
||||
.select_related("attributed_to")
|
||||
.prefetch_related(
|
||||
|
|
@ -130,6 +105,7 @@ class ManageAlbumViewSet(
|
|||
):
|
||||
queryset = (
|
||||
music_models.Album.objects.all()
|
||||
.distinct()
|
||||
.order_by("-id")
|
||||
.select_related("attributed_to", "artist")
|
||||
.prefetch_related("tracks")
|
||||
|
|
@ -164,6 +140,7 @@ class ManageTrackViewSet(
|
|||
):
|
||||
queryset = (
|
||||
music_models.Track.objects.all()
|
||||
.distinct()
|
||||
.order_by("-id")
|
||||
.select_related("attributed_to", "artist", "album__artist")
|
||||
.annotate(uploads_count=Count("uploads"))
|
||||
|
|
@ -196,6 +173,96 @@ class ManageTrackViewSet(
|
|||
return response.Response(result, status=200)
|
||||
|
||||
|
||||
class ManageLibraryViewSet(
|
||||
mixins.ListModelMixin,
|
||||
mixins.RetrieveModelMixin,
|
||||
mixins.DestroyModelMixin,
|
||||
viewsets.GenericViewSet,
|
||||
):
|
||||
lookup_field = "uuid"
|
||||
queryset = (
|
||||
music_models.Library.objects.all()
|
||||
.distinct()
|
||||
.order_by("-id")
|
||||
.select_related("actor")
|
||||
.annotate(
|
||||
followers_count=Count("received_follows", distinct=True),
|
||||
_uploads_count=Count("uploads", distinct=True),
|
||||
)
|
||||
)
|
||||
serializer_class = serializers.ManageLibrarySerializer
|
||||
filterset_class = filters.ManageLibraryFilterSet
|
||||
required_scope = "instance:libraries"
|
||||
|
||||
@rest_decorators.action(methods=["get"], detail=True)
|
||||
def stats(self, request, *args, **kwargs):
|
||||
library = self.get_object()
|
||||
uploads = library.uploads.all()
|
||||
tracks = uploads.values_list("track", flat=True).distinct()
|
||||
albums = (
|
||||
music_models.Track.objects.filter(pk__in=tracks)
|
||||
.values_list("album", flat=True)
|
||||
.distinct()
|
||||
)
|
||||
artists = set(
|
||||
music_models.Album.objects.filter(pk__in=albums).values_list(
|
||||
"artist", flat=True
|
||||
)
|
||||
) | set(
|
||||
music_models.Track.objects.filter(pk__in=tracks).values_list(
|
||||
"artist", flat=True
|
||||
)
|
||||
)
|
||||
|
||||
data = {
|
||||
"uploads": uploads.count(),
|
||||
"followers": library.received_follows.count(),
|
||||
"tracks": tracks.count(),
|
||||
"albums": albums.count(),
|
||||
"artists": len(artists),
|
||||
}
|
||||
data.update(get_media_stats(uploads.all()))
|
||||
return response.Response(data, status=200)
|
||||
|
||||
@rest_decorators.action(methods=["post"], detail=False)
|
||||
def action(self, request, *args, **kwargs):
|
||||
queryset = self.get_queryset()
|
||||
serializer = serializers.ManageTrackActionSerializer(
|
||||
request.data, queryset=queryset
|
||||
)
|
||||
serializer.is_valid(raise_exception=True)
|
||||
result = serializer.save()
|
||||
return response.Response(result, status=200)
|
||||
|
||||
|
||||
class ManageUploadViewSet(
|
||||
mixins.ListModelMixin,
|
||||
mixins.RetrieveModelMixin,
|
||||
mixins.DestroyModelMixin,
|
||||
viewsets.GenericViewSet,
|
||||
):
|
||||
lookup_field = "uuid"
|
||||
queryset = (
|
||||
music_models.Upload.objects.all()
|
||||
.distinct()
|
||||
.order_by("-id")
|
||||
.select_related("library__actor", "track__artist", "track__album__artist")
|
||||
)
|
||||
serializer_class = serializers.ManageUploadSerializer
|
||||
filterset_class = filters.ManageUploadFilterSet
|
||||
required_scope = "instance:libraries"
|
||||
|
||||
@rest_decorators.action(methods=["post"], detail=False)
|
||||
def action(self, request, *args, **kwargs):
|
||||
queryset = self.get_queryset()
|
||||
serializer = serializers.ManageTrackActionSerializer(
|
||||
request.data, queryset=queryset
|
||||
)
|
||||
serializer.is_valid(raise_exception=True)
|
||||
result = serializer.save()
|
||||
return response.Response(result, status=200)
|
||||
|
||||
|
||||
class ManageUserViewSet(
|
||||
mixins.ListModelMixin,
|
||||
mixins.RetrieveModelMixin,
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue