mirror of
https://github.com/spl0k/supysonic.git
synced 2025-01-12 03:06:18 +00:00
Porting supysonic.api.playlists
This commit is contained in:
parent
50b98e641a
commit
6179daf4ce
@ -9,30 +9,29 @@ import uuid
|
|||||||
|
|
||||||
from flask import request
|
from flask import request
|
||||||
|
|
||||||
from ..db import Playlist, User, Track
|
from ..db import Playlist, User, Track, db
|
||||||
|
|
||||||
from . import get_entity, api_routing
|
from . import get_entity, api_routing
|
||||||
from .exceptions import Forbidden, MissingParameter, NotFound
|
from .exceptions import Forbidden, MissingParameter
|
||||||
|
|
||||||
|
|
||||||
@api_routing("/getPlaylists")
|
@api_routing("/getPlaylists")
|
||||||
def list_playlists():
|
def list_playlists():
|
||||||
query = Playlist.select(
|
query = (
|
||||||
lambda p: p.user.id == request.user.id or p.public
|
Playlist.select()
|
||||||
).order_by(Playlist.name)
|
.orwhere(Playlist.user == request.user, Playlist.public)
|
||||||
|
.order_by(Playlist.name)
|
||||||
|
)
|
||||||
|
|
||||||
username = request.values.get("username")
|
username = request.values.get("username")
|
||||||
if username:
|
if username:
|
||||||
if not request.user.admin:
|
if not request.user.admin:
|
||||||
raise Forbidden()
|
raise Forbidden()
|
||||||
|
|
||||||
|
# get rather than join in the following query to raise an exception if the
|
||||||
|
# requested user doesn't exist
|
||||||
user = User.get(name=username)
|
user = User.get(name=username)
|
||||||
if user is None:
|
query = Playlist.select().where(Playlist.user == user).order_by(Playlist.name)
|
||||||
raise NotFound("User")
|
|
||||||
|
|
||||||
query = Playlist.select(lambda p: p.user.name == username).order_by(
|
|
||||||
Playlist.name
|
|
||||||
)
|
|
||||||
|
|
||||||
return request.formatter(
|
return request.formatter(
|
||||||
"playlists",
|
"playlists",
|
||||||
@ -43,7 +42,7 @@ def list_playlists():
|
|||||||
@api_routing("/getPlaylist")
|
@api_routing("/getPlaylist")
|
||||||
def show_playlist():
|
def show_playlist():
|
||||||
res = get_entity(Playlist)
|
res = get_entity(Playlist)
|
||||||
if res.user.id != request.user.id and not res.public and not request.user.admin:
|
if res.user != request.user and not res.public and not request.user.admin:
|
||||||
raise Forbidden()
|
raise Forbidden()
|
||||||
|
|
||||||
info = res.as_subsonic_playlist(request.user)
|
info = res.as_subsonic_playlist(request.user)
|
||||||
@ -54,6 +53,7 @@ def show_playlist():
|
|||||||
|
|
||||||
|
|
||||||
@api_routing("/createPlaylist")
|
@api_routing("/createPlaylist")
|
||||||
|
@db.atomic()
|
||||||
def create_playlist():
|
def create_playlist():
|
||||||
playlist_id, name = map(request.values.get, ("playlistId", "name"))
|
playlist_id, name = map(request.values.get, ("playlistId", "name"))
|
||||||
# songId actually doesn't seem to be required
|
# songId actually doesn't seem to be required
|
||||||
@ -63,14 +63,14 @@ def create_playlist():
|
|||||||
if playlist_id:
|
if playlist_id:
|
||||||
playlist = Playlist[playlist_id]
|
playlist = Playlist[playlist_id]
|
||||||
|
|
||||||
if playlist.user.id != request.user.id and not request.user.admin:
|
if playlist.user != request.user and not request.user.admin:
|
||||||
raise Forbidden()
|
raise Forbidden()
|
||||||
|
|
||||||
playlist.clear()
|
playlist.clear()
|
||||||
if name:
|
if name:
|
||||||
playlist.name = name
|
playlist.name = name
|
||||||
elif name:
|
elif name:
|
||||||
playlist = Playlist(user=request.user, name=name)
|
playlist = Playlist.create(user=request.user, name=name)
|
||||||
else:
|
else:
|
||||||
raise MissingParameter("playlistId or name")
|
raise MissingParameter("playlistId or name")
|
||||||
|
|
||||||
@ -78,6 +78,7 @@ def create_playlist():
|
|||||||
sid = uuid.UUID(sid)
|
sid = uuid.UUID(sid)
|
||||||
track = Track[sid]
|
track = Track[sid]
|
||||||
playlist.add(track)
|
playlist.add(track)
|
||||||
|
playlist.save()
|
||||||
|
|
||||||
return request.formatter.empty
|
return request.formatter.empty
|
||||||
|
|
||||||
@ -85,17 +86,17 @@ def create_playlist():
|
|||||||
@api_routing("/deletePlaylist")
|
@api_routing("/deletePlaylist")
|
||||||
def delete_playlist():
|
def delete_playlist():
|
||||||
res = get_entity(Playlist)
|
res = get_entity(Playlist)
|
||||||
if res.user.id != request.user.id and not request.user.admin:
|
if res.user != request.user and not request.user.admin:
|
||||||
raise Forbidden()
|
raise Forbidden()
|
||||||
|
|
||||||
res.delete()
|
res.delete_instance()
|
||||||
return request.formatter.empty
|
return request.formatter.empty
|
||||||
|
|
||||||
|
|
||||||
@api_routing("/updatePlaylist")
|
@api_routing("/updatePlaylist")
|
||||||
def update_playlist():
|
def update_playlist():
|
||||||
res = get_entity(Playlist, "playlistId")
|
res = get_entity(Playlist, "playlistId")
|
||||||
if res.user.id != request.user.id and not request.user.admin:
|
if res.user != request.user and not request.user.admin:
|
||||||
raise Forbidden()
|
raise Forbidden()
|
||||||
|
|
||||||
playlist = res
|
playlist = res
|
||||||
@ -119,5 +120,6 @@ def update_playlist():
|
|||||||
playlist.add(track)
|
playlist.add(track)
|
||||||
|
|
||||||
playlist.remove_at_indexes(to_remove)
|
playlist.remove_at_indexes(to_remove)
|
||||||
|
playlist.save()
|
||||||
|
|
||||||
return request.formatter.empty
|
return request.formatter.empty
|
||||||
|
@ -1,15 +1,13 @@
|
|||||||
# This file is part of Supysonic.
|
# This file is part of Supysonic.
|
||||||
# Supysonic is a Python implementation of the Subsonic server API.
|
# Supysonic is a Python implementation of the Subsonic server API.
|
||||||
#
|
#
|
||||||
# Copyright (C) 2017-2018 Alban 'spl0k' Féron
|
# Copyright (C) 2017-2022 Alban 'spl0k' Féron
|
||||||
#
|
#
|
||||||
# Distributed under terms of the GNU AGPLv3 license.
|
# Distributed under terms of the GNU AGPLv3 license.
|
||||||
|
|
||||||
import unittest
|
import unittest
|
||||||
import uuid
|
import uuid
|
||||||
|
|
||||||
from pony.orm import db_session
|
|
||||||
|
|
||||||
from supysonic.db import Folder, Artist, Album, Track, Playlist, User
|
from supysonic.db import Folder, Artist, Album, Track, Playlist, User
|
||||||
|
|
||||||
from .apitestbase import ApiTestBase
|
from .apitestbase import ApiTestBase
|
||||||
@ -19,14 +17,13 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
def setUp(self):
|
def setUp(self):
|
||||||
super().setUp()
|
super().setUp()
|
||||||
|
|
||||||
with db_session:
|
root = Folder.create(root=True, name="Root folder", path="tests/assets")
|
||||||
root = Folder(root=True, name="Root folder", path="tests/assets")
|
artist = Artist.create(name="Artist")
|
||||||
artist = Artist(name="Artist")
|
album = Album.create(name="Album", artist=artist)
|
||||||
album = Album(name="Album", artist=artist)
|
|
||||||
|
|
||||||
songs = {}
|
songs = {}
|
||||||
for num, song in enumerate(["One", "Two", "Three", "Four"]):
|
for num, song in enumerate(["One", "Two", "Three", "Four"]):
|
||||||
track = Track(
|
track = Track.create(
|
||||||
disc=1,
|
disc=1,
|
||||||
number=num,
|
number=num,
|
||||||
title=song,
|
title=song,
|
||||||
@ -43,17 +40,22 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
|
|
||||||
users = {u.name: u for u in User.select()}
|
users = {u.name: u for u in User.select()}
|
||||||
|
|
||||||
playlist = Playlist(user=users["alice"], name="Alice's")
|
playlist = Playlist.create(user=users["alice"], name="Alice's")
|
||||||
playlist.add(songs["One"])
|
playlist.add(songs["One"])
|
||||||
playlist.add(songs["Three"])
|
playlist.add(songs["Three"])
|
||||||
|
playlist.save()
|
||||||
|
|
||||||
playlist = Playlist(user=users["alice"], public=True, name="Alice's public")
|
playlist = Playlist.create(
|
||||||
|
user=users["alice"], public=True, name="Alice's public"
|
||||||
|
)
|
||||||
playlist.add(songs["One"])
|
playlist.add(songs["One"])
|
||||||
playlist.add(songs["Two"])
|
playlist.add(songs["Two"])
|
||||||
|
playlist.save()
|
||||||
|
|
||||||
playlist = Playlist(user=users["bob"], name="Bob's")
|
playlist = Playlist.create(user=users["bob"], name="Bob's")
|
||||||
playlist.add(songs["Two"])
|
playlist.add(songs["Two"])
|
||||||
playlist.add(songs["Four"])
|
playlist.add(songs["Four"])
|
||||||
|
playlist.save()
|
||||||
|
|
||||||
def test_get_playlists(self):
|
def test_get_playlists(self):
|
||||||
# get own playlists
|
# get own playlists
|
||||||
@ -99,8 +101,12 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
self._make_request("getPlaylist", {"id": str(uuid.uuid4())}, error=70)
|
self._make_request("getPlaylist", {"id": str(uuid.uuid4())}, error=70)
|
||||||
|
|
||||||
# other's private from non admin
|
# other's private from non admin
|
||||||
with db_session:
|
playlist = (
|
||||||
playlist = Playlist.get(lambda p: not p.public and p.user.name == "alice")
|
Playlist.select()
|
||||||
|
.join(User)
|
||||||
|
.where(~Playlist.public, User.name == "alice")
|
||||||
|
.get()
|
||||||
|
)
|
||||||
self._make_request(
|
self._make_request(
|
||||||
"getPlaylist", {"u": "bob", "p": "B0b", "id": str(playlist.id)}, error=50
|
"getPlaylist", {"u": "bob", "p": "B0b", "id": str(playlist.id)}, error=50
|
||||||
)
|
)
|
||||||
@ -166,7 +172,6 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
# create more useful playlist
|
# create more useful playlist
|
||||||
with db_session:
|
|
||||||
songs = {s.title: str(s.id) for s in Track.select()}
|
songs = {s.title: str(s.id) for s in Track.select()}
|
||||||
self._make_request(
|
self._make_request(
|
||||||
"createPlaylist",
|
"createPlaylist",
|
||||||
@ -176,7 +181,6 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
},
|
},
|
||||||
skip_post=True,
|
skip_post=True,
|
||||||
)
|
)
|
||||||
with db_session:
|
|
||||||
playlist = Playlist.get(name="songs")
|
playlist = Playlist.get(name="songs")
|
||||||
self.assertIsNotNone(playlist)
|
self.assertIsNotNone(playlist)
|
||||||
rv, child = self._make_request(
|
rv, child = self._make_request(
|
||||||
@ -201,7 +205,6 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
self.assertEqual(self._xpath(child, "count(./entry)"), 1)
|
self.assertEqual(self._xpath(child, "count(./entry)"), 1)
|
||||||
self.assertEqual(child[0].get("title"), "Two")
|
self.assertEqual(child[0].get("title"), "Two")
|
||||||
|
|
||||||
@db_session
|
|
||||||
def assertPlaylistCountEqual(self, count):
|
def assertPlaylistCountEqual(self, count):
|
||||||
self.assertEqual(Playlist.select().count(), count)
|
self.assertEqual(Playlist.select().count(), count)
|
||||||
|
|
||||||
@ -212,8 +215,7 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
self._make_request("deletePlaylist", {"id": str(uuid.uuid4())}, error=70)
|
self._make_request("deletePlaylist", {"id": str(uuid.uuid4())}, error=70)
|
||||||
|
|
||||||
# delete unowned when not admin
|
# delete unowned when not admin
|
||||||
with db_session:
|
playlist = Playlist.select().join(User).where(User.name == "alice").first()
|
||||||
playlist = Playlist.select(lambda p: p.user.name == "alice").first()
|
|
||||||
self._make_request(
|
self._make_request(
|
||||||
"deletePlaylist", {"u": "bob", "p": "B0b", "id": str(playlist.id)}, error=50
|
"deletePlaylist", {"u": "bob", "p": "B0b", "id": str(playlist.id)}, error=50
|
||||||
)
|
)
|
||||||
@ -226,8 +228,7 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
self.assertPlaylistCountEqual(2)
|
self.assertPlaylistCountEqual(2)
|
||||||
|
|
||||||
# delete unowned when admin
|
# delete unowned when admin
|
||||||
with db_session:
|
playlist = Playlist.select().join(User).where(User.name == "bob").get()
|
||||||
playlist = Playlist.get(lambda p: p.user.name == "bob")
|
|
||||||
self._make_request("deletePlaylist", {"id": str(playlist.id)}, skip_post=True)
|
self._make_request("deletePlaylist", {"id": str(playlist.id)}, skip_post=True)
|
||||||
self.assertPlaylistCountEqual(1)
|
self.assertPlaylistCountEqual(1)
|
||||||
|
|
||||||
@ -238,9 +239,10 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
"updatePlaylist", {"playlistId": str(uuid.uuid4())}, error=70
|
"updatePlaylist", {"playlistId": str(uuid.uuid4())}, error=70
|
||||||
)
|
)
|
||||||
|
|
||||||
with db_session:
|
|
||||||
playlist = (
|
playlist = (
|
||||||
Playlist.select(lambda p: p.user.name == "alice")
|
Playlist.select()
|
||||||
|
.join(User)
|
||||||
|
.where(User.name == "alice")
|
||||||
.order_by(Playlist.created)
|
.order_by(Playlist.created)
|
||||||
.first()
|
.first()
|
||||||
)
|
)
|
||||||
@ -288,7 +290,6 @@ class PlaylistTestCase(ApiTestBase):
|
|||||||
self.assertEqual(self._xpath(child, "count(./entry)"), 1)
|
self.assertEqual(self._xpath(child, "count(./entry)"), 1)
|
||||||
self.assertEqual(self._find(child, "./entry").get("title"), "Three")
|
self.assertEqual(self._find(child, "./entry").get("title"), "Three")
|
||||||
|
|
||||||
with db_session:
|
|
||||||
songs = {s.title: str(s.id) for s in Track.select()}
|
songs = {s.title: str(s.id) for s in Track.select()}
|
||||||
|
|
||||||
self._make_request(
|
self._make_request(
|
||||||
|
Loading…
Reference in New Issue
Block a user