From 72dc85ac74b060cb531fcc6a4e606532df80e0a9 Mon Sep 17 00:00:00 2001 From: evilchili Date: Mon, 5 Dec 2022 01:06:57 -0800 Subject: [PATCH] add tests --- .test_env | 6 -- groove/path.py | 43 +++++++---- groove/webserver/themes.py | 26 ++++--- groove/webserver/webserver.py | 69 +++++++++--------- pytest.ini | 2 +- test/conftest.py | 24 +++++- test/fixtures/env | 24 ++++++ ...Guns Blazing (Drums of Death, Part 1).flac | 1 + test/fixtures/media/foo.mp3 | 0 test/fixtures/static/favicon.ico | 1 + test/fixtures/themes/alt_theme/README.md | 0 .../fixtures/themes/alt_theme/static/test.css | 0 .../themes/alt_theme/templates/playlist.tpl | 0 test/fixtures/themes/default_theme/README.md | 11 +++ .../themes/default_theme/static/test.css | 1 + .../default_theme/templates/playlist.tpl | 0 test/test.db | Bin 0 -> 28672 bytes test/test_path.py | 13 ++++ test/test_requests.py | 19 +++++ test/test_scanner.py | 4 +- test/test_themes.py | 16 ++++ test/test_webserver.py | 56 +++++++++++--- 22 files changed, 236 insertions(+), 80 deletions(-) delete mode 100644 .test_env create mode 100644 test/fixtures/env create mode 100644 test/fixtures/media/UNKLE/Psyence Fiction/01 Guns Blazing (Drums of Death, Part 1).flac create mode 100644 test/fixtures/media/foo.mp3 create mode 100644 test/fixtures/static/favicon.ico create mode 100644 test/fixtures/themes/alt_theme/README.md create mode 100644 test/fixtures/themes/alt_theme/static/test.css create mode 100644 test/fixtures/themes/alt_theme/templates/playlist.tpl create mode 100644 test/fixtures/themes/default_theme/README.md create mode 100644 test/fixtures/themes/default_theme/static/test.css create mode 100644 test/fixtures/themes/default_theme/templates/playlist.tpl create mode 100644 test/test.db create mode 100644 test/test_path.py create mode 100644 test/test_requests.py create mode 100644 test/test_themes.py diff --git a/.test_env b/.test_env deleted file mode 100644 index 7313ff4..0000000 --- a/.test_env +++ /dev/null @@ -1,6 +0,0 @@ -HOST=127.0.0.1 -DEBUG=1 -USERNAME=test_username -PASSWORD=test_password -MEDIA_ROOT=/test -MEDIA_GLOB=*.flac,*.mp3 diff --git a/groove/path.py b/groove/path.py index 7722fef..f0c6309 100644 --- a/groove/path.py +++ b/groove/path.py @@ -1,7 +1,8 @@ +import logging import os from pathlib import Path -from groove.exceptions import ConfigurationError, ThemeMissingException +from groove.exceptions import ConfigurationError, ThemeMissingException, ThemeConfigurationError _setup_hint = "You may be able to solve this error by running 'groove setup'." _reinstall_hint = "You might need to reinstall Groove On Demand to fix this error." @@ -12,11 +13,12 @@ def root(): if not path: raise ConfigurationError(f"GROOVE_ON_DEMAND_ROOT is not defined in your environment.\n\n{_setup_hint}") path = Path(path).expanduser() - if not path.exists or not path.is_dir: + if not path.exists() or not path.is_dir(): raise ConfigurationError( "The Groove on Demand root directory (GROOVE_ON_DEMAND_ROOT) " f"does not exist or isn't a directory.\n\n{_reinstall_hint}" ) + logging.debug(f"Root is {path}") return Path(path) @@ -24,47 +26,64 @@ def media_root(): path = os.environ.get('MEDIA_ROOT', None) if not path: raise ConfigurationError(f"MEDIA_ROOT is not defined in your environment.\n\n{_setup_hint}") - path =Path(path).expanduser() - if not path.exists and path.is_dir: + path = Path(path).expanduser() + if not path.exists() or not path.is_dir(): raise ConfigurationError( "The media_root directory (MEDIA_ROOT) doesn't exist, or isn't a directory.\n\n{_setup_hint}" ) + logging.debug(f"Media root is {path}") return path def media(relpath): - return themes_root() / Path(relpath) + path = media_root() / Path(relpath) + return path def static_root(): dirname = os.environ.get('STATIC_PATH', 'static') path = root() / Path(dirname) - if not path.exists or not path.is_dir: + if not path.exists() or not path.is_dir(): raise ConfigurationError( f"The static assets directory {dirname} (STATIC_PATH) " f"doesn't exist, or isn't a directory.\n\n{_reinstall_hint}" ) + logging.debug(f"Static root is {path}") return path -def static(relpath): - return static_root() / Path(relpath) +def static(relpath, theme=None): + if theme: + root = theme.path / Path('static') + if not root.is_dir(): + raise ThemeConfigurationError( + f"The themes directory {relpath} (THEMES_PATH) " + f"doesn't contain a 'static' directory." + ) + path = root / Path(relpath) + logging.debug(f"Checking for {path}") + if path.exists(): + return path + path = static_root() / Path(relpath) + logging.debug(f"Defaulting to {path}") + return path def themes_root(): dirname = os.environ.get('THEMES_PATH', 'themes') path = root() / Path(dirname) - if not path.exists or not path.is_dir: + if not path.exists() or not path.is_dir(): raise ConfigurationError( f"The themes directory {dirname} (THEMES_PATH) " f"doesn't exist, or isn't a directory.\n\n{_reinstall_hint}" ) + logging.debug(f"Themes root is {path}") return path def theme(name): path = themes_root() / Path(name) - if not path.exists or not path.is_dir: + if not path.exists() or not path.is_dir(): available = ','.join(available_themes) raise ThemeMissingException( f"A theme directory named {name} does not exist or isn't a directory. " @@ -74,10 +93,6 @@ def theme(name): return path -def theme_static(relpath): - return Path('static') / Path(relpath) - - def theme_template(template_name): return Path('templates') / Path(f"{template_name}.tpl") diff --git a/groove/webserver/themes.py b/groove/webserver/themes.py index ae5bfc8..454c569 100644 --- a/groove/webserver/themes.py +++ b/groove/webserver/themes.py @@ -8,32 +8,38 @@ from groove.exceptions import ThemeConfigurationError, ConfigurationError import groove.path -Theme = namedtuple('Theme', 'path,author,author_link,version,about') +Theme = namedtuple('Theme', 'name,path,author,author_link,version,about') def load_theme(name=None): - name = os.environ.get('DEFAULT_THEME', name) - if not name: + name = name or os.environ.get('DEFAULT_THEME', None) + if not name: # pragma: no cover raise ConfigurationError( "It seems like DEFAULT_THEME is not set in your current environment.\n" "Running 'groove setup' may help you fix this problem." ) theme_path = groove.path.theme(name) theme_info = _get_theme_info(theme_path) - return Theme( - path=theme_path, - **theme_info - ) + try: + return Theme( + name=name, + path=theme_path, + **theme_info + ) + except TypeError: + raise ThemeConfigurationError(f"The {name} them is misconfigured. Does the README.md contain a credits secton?") def _get_theme_info(theme_path): readme = theme_path / Path('README.md') - if not readme.exists: + if not readme.exists: # pragma: no cover raise ThemeConfigurationError( "The theme is missing a required file: README.md.\n" "Refer to the Groove On Demand documentation for help creating themes." ) - config = {'about': ''} + config = { + 'about': '', + } with readme.open() as fh: in_credits = False in_block = False @@ -55,7 +61,7 @@ def _get_theme_info(theme_path): key = key.strip() value = value.strip() except ValueError: - logging.warn(f"Could not parse credits line: {line}") + logging.warning(f"Could not parse credits line: {line}") continue logging.debug(f"Setting theme '{key}' to '{value}'.") config[key] = value diff --git a/groove/webserver/webserver.py b/groove/webserver/webserver.py index 95b5488..e3b88ff 100644 --- a/groove/webserver/webserver.py +++ b/groove/webserver/webserver.py @@ -5,6 +5,7 @@ import os import bottle from bottle import HTTPResponse, template, static_file from bottle.ext import sqlalchemy +from sqlalchemy.exc import NoResultFound, MultipleResultsFound import groove.db from groove.auth import is_authenticated @@ -54,54 +55,32 @@ def index(): return "Groovy." -@server.route('/build') -@bottle.auth_basic(is_authenticated) -def build(): - return "Authenticated. Groovy." - - @server.route('/static/') -def server_static(filepath): +def serve_static(filepath): theme = themes.load_theme() - asset = theme.path / groove.path.theme_static(filepath) - if asset.exists(): - root = asset.parent - else: - root = groove.path.static_root() - asset = groove.path.static(filepath) - if asset.is_dir(): - logging.warning("Asset {asset} is a directory; returning 404.") - return HTTPResponse(status=404, body="Not found.") - logging.debug(f"Serving asset {asset.name} from {root}") - return static_file(asset.name, root=root) - - -@bottle.auth_basic(is_authenticated) -@server.route('/build/search/playlist') -def search_playlist(slug, db): - playlist = Playlist(slug=slug, session=db, create_ok=False) - response = json.dumps(playlist.as_dict) - logging.debug(response) - return HTTPResponse(status=200, content_type='application/json', body=response) + path = groove.path.static(filepath, theme=theme) + logging.debug(f"Serving asset {path.name} from {path.parent}") + return static_file(path.name, root=path.parent) @server.route('/track//') def serve_track(request, track_id, db): expected = requests.encode([track_id], '/track') - if not requests.verify(request, expected): + if not requests.verify(request, expected): # pragma: no cover return HTTPResponse(status=404, body="Not found") - track_id = int(track_id) - track = db.query(groove.db.track).filter( - groove.db.track.c.id == track_id - ).one() + try: + track_id = int(track_id) + track = db.query(groove.db.track).filter( + groove.db.track.c.id == track_id + ).one() + except (NoResultFound, MultipleResultsFound): + return HTTPResponse(status=404, body="Not found") path = groove.path.media(track['relpath']) - if path.exists: - return static_file(path.name, root=path.parent) - else: - return HTTPResponse(status=404, body="Not found") + logging.debug(f"Service track {path.name} from {path.parent}") + return static_file(path.name, root=path.parent) @server.route('/playlist/') @@ -123,3 +102,21 @@ def serve_playlist(slug, db): entry['url'] = f"/track/{sig}/{entry['track_id']}" return serve('playlist', playlist=pl) + + +@server.route('/build') +@bottle.auth_basic(is_authenticated) +def build(): + return "Authenticated. Groovy." + + +@bottle.auth_basic(is_authenticated) +@server.route('/build/search/playlist/') +def search_playlist(slug, db): + playlist = Playlist(slug=slug, session=db, create_ok=False).load() + if not playlist.record: + logging.debug(f"Playist {slug} doesn't exist.") + body = {} + else: + body = json.dumps(playlist.as_dict) + return HTTPResponse(status=200, content_type='application/json', body=body) diff --git a/pytest.ini b/pytest.ini index 10d660a..62a6234 100644 --- a/pytest.ini +++ b/pytest.ini @@ -1,5 +1,5 @@ [pytest] env_override_existing_values = 1 -env_files = .test_env +env_files = test/fixtures/env log_cli_level = DEBUG addopts = --cov=groove/ --cov-report=term-missing diff --git a/test/conftest.py b/test/conftest.py index c906c61..d62e12a 100644 --- a/test/conftest.py +++ b/test/conftest.py @@ -1,4 +1,8 @@ import pytest +import os + +from pathlib import Path +from dotenv import load_dotenv import groove.db @@ -6,6 +10,20 @@ from sqlalchemy import create_engine, insert from sqlalchemy.orm import sessionmaker +@pytest.fixture(autouse=True, scope='function') +def env(): + root = Path(__file__).parent / Path('fixtures') + load_dotenv(Path('test/fixtures/env')) + os.environ['GROOVE_ON_DEMAND_ROOT'] = str(root) + os.environ['MEDIA_ROOT'] = str(root / Path('media')) + return os.environ + + +@pytest.fixture(scope='function') +def auth(): + return (os.environ.get('USERNAME'), os.environ.get('PASSWORD')) + + @pytest.fixture(scope='function') def in_memory_engine(): return create_engine('sqlite:///:memory:', future=True) @@ -32,9 +50,9 @@ def db(in_memory_db): # create tracks query = insert(groove.db.track) in_memory_db.execute(query, [ - {'id': 1, 'relpath': '/UNKLE/Psyence Fiction/01 Guns Blazing (Drums of Death, Part 1).flac'}, - {'id': 2, 'relpath': '/UNKLE/Psyence Fiction/02 UNKLE (Main Title Theme).flac'}, - {'id': 3, 'relpath': '/UNKLE/Psyence Fiction/03 Bloodstain.flac'} + {'id': 1, 'relpath': 'UNKLE/Psyence Fiction/01 Guns Blazing (Drums of Death, Part 1).flac'}, + {'id': 2, 'relpath': 'UNKLE/Psyence Fiction/02 UNKLE (Main Title Theme).flac'}, + {'id': 3, 'relpath': 'UNKLE/Psyence Fiction/03 Bloodstain.flac'} ]) # create playlists diff --git a/test/fixtures/env b/test/fixtures/env new file mode 100644 index 0000000..3ed12aa --- /dev/null +++ b/test/fixtures/env @@ -0,0 +1,24 @@ +# Will be overwritten by test setup +GROOVE_ON_DEMAND_ROOT=. +MEDIA_ROOT=. + +# where to store the database +DATABASE_PATH=test.db + +# Admin user credentials +USERNAME=test_username +PASSWORD=test_password + +# Web interface configuration +HOST=127.0.0.1 +PORT=2323 +THEMES_PATH=themes +STATIC_PATH=static +DEFAULT_THEME=default_theme +SECRET_KEY=fnord + +# Media scanner configuration +MEDIA_GLOB=*.mp3,*.flac,*.m4a + +# Set this value to enable debugging +DEBUG=1 diff --git a/test/fixtures/media/UNKLE/Psyence Fiction/01 Guns Blazing (Drums of Death, Part 1).flac b/test/fixtures/media/UNKLE/Psyence Fiction/01 Guns Blazing (Drums of Death, Part 1).flac new file mode 100644 index 0000000..c8f90a5 --- /dev/null +++ b/test/fixtures/media/UNKLE/Psyence Fiction/01 Guns Blazing (Drums of Death, Part 1).flac @@ -0,0 +1 @@ +DRUMS OF DEATH YALL diff --git a/test/fixtures/media/foo.mp3 b/test/fixtures/media/foo.mp3 new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/static/favicon.ico b/test/fixtures/static/favicon.ico new file mode 100644 index 0000000..a4d9b91 --- /dev/null +++ b/test/fixtures/static/favicon.ico @@ -0,0 +1 @@ +favicon.ico diff --git a/test/fixtures/themes/alt_theme/README.md b/test/fixtures/themes/alt_theme/README.md new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/themes/alt_theme/static/test.css b/test/fixtures/themes/alt_theme/static/test.css new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/themes/alt_theme/templates/playlist.tpl b/test/fixtures/themes/alt_theme/templates/playlist.tpl new file mode 100644 index 0000000..e69de29 diff --git a/test/fixtures/themes/default_theme/README.md b/test/fixtures/themes/default_theme/README.md new file mode 100644 index 0000000..50ab655 --- /dev/null +++ b/test/fixtures/themes/default_theme/README.md @@ -0,0 +1,11 @@ +# Default Theme Fixture + +This directory is a test fixture used for verifying theme handling. + +## Credits + +``` +author: Theme Author +author_link: link to my soundcloud +version: 1.3 +``` diff --git a/test/fixtures/themes/default_theme/static/test.css b/test/fixtures/themes/default_theme/static/test.css new file mode 100644 index 0000000..05a0165 --- /dev/null +++ b/test/fixtures/themes/default_theme/static/test.css @@ -0,0 +1 @@ +/* test.css */ diff --git a/test/fixtures/themes/default_theme/templates/playlist.tpl b/test/fixtures/themes/default_theme/templates/playlist.tpl new file mode 100644 index 0000000..e69de29 diff --git a/test/test.db b/test/test.db new file mode 100644 index 0000000000000000000000000000000000000000..8b492042518a62310afd46e910f6d6423baea81e GIT binary patch literal 28672 zcmeI%zi-n(6u@!2`9-6qt;FCd(k)1_)FL5fBtp2BKyav=Kq>~yG^SX%Nl09&fhn-@ z2X|n?ca4KX$;M=TE&0dU@7aEz=f%3ylTH{1;^TOF>Br)+v2U2B@l*)IFiOc?up%>yN5eBwO^YOIFA4V2q1s}0tg_000Id7 zUxDvtsbX1{`NNO>4^a?|;^|y($}Lwmds6h8&pJ}*Q=(Dbj;H?lXVG?g@>sgU>Gp)v z?{p4Cb$b%|^C+CfUO4>cb)A{HsB+%8?bl8BOuUk3qLI15fzZ*miaF}KvVH8Rm~4nG zTzMp2>9pjjNK-{49NtMvAH0@vTZam-+IG2YJ+Vw991Vl(*_SB!4fOo0c&x{sF5P>i zcj_nbo@&7RV%a*Zo8NRucpAP{61Q0*~0R#|0009ILKmdXK3o!qm{~l8z0tg_0 z00IagfB*srAbVNZAb