mirror of
https://github.com/tcsenpai/swingmusic.git
synced 2025-06-08 12:15:39 +00:00

- watch route params instead of route object in folderview - move to script setup on album view - use album as a reactive object instead of refs - use axios instead of fetch to get album data - improve clickable areas on songItem - move album requests to POST
175 lines
3.9 KiB
Python
175 lines
3.9 KiB
Python
"""
|
|
This module contains mimi functions for the server.
|
|
"""
|
|
|
|
import os
|
|
import threading
|
|
import time
|
|
from typing import List
|
|
import requests
|
|
import colorgram
|
|
|
|
from io import BytesIO
|
|
|
|
from PIL import Image
|
|
|
|
from app import instances
|
|
from app import functions
|
|
from app import watchdoge
|
|
from app import models
|
|
|
|
home_dir = os.path.expanduser("~") + "/"
|
|
app_dir = os.path.join(home_dir, ".musicx")
|
|
LAST_FM_API_KEY = "762db7a44a9e6fb5585661f5f2bdf23a"
|
|
|
|
|
|
def background(func):
|
|
"""
|
|
a threading decorator
|
|
use @background above the function you want to run in the background
|
|
"""
|
|
|
|
def background_func(*a, **kw):
|
|
threading.Thread(target=func, args=a, kwargs=kw).start()
|
|
|
|
return background_func
|
|
|
|
|
|
@background
|
|
def reindex_tracks():
|
|
"""
|
|
Checks for new songs every 5 minutes.
|
|
"""
|
|
flag = False
|
|
|
|
while flag is False:
|
|
functions.populate()
|
|
functions.populate_images()
|
|
time.sleep(300)
|
|
|
|
|
|
@background
|
|
def start_watchdog():
|
|
"""
|
|
Starts the file watcher.
|
|
"""
|
|
watchdoge.watch.run()
|
|
|
|
|
|
def run_fast_scandir(_dir: str, ext: list):
|
|
"""
|
|
Scans a directory for files with a specific extension. Returns a list of files and folders in the directory.
|
|
"""
|
|
|
|
subfolders = []
|
|
files = []
|
|
|
|
for f in os.scandir(_dir):
|
|
if f.is_dir() and not f.name.startswith("."):
|
|
subfolders.append(f.path)
|
|
if f.is_file():
|
|
if os.path.splitext(f.name)[1].lower() in ext:
|
|
files.append(f.path)
|
|
|
|
for _dir in list(subfolders):
|
|
sf, f = run_fast_scandir(_dir, ext)
|
|
subfolders.extend(sf)
|
|
files.extend(f)
|
|
|
|
return subfolders, files
|
|
|
|
|
|
def remove_duplicates(tracklist: List[models.Track]) -> List[models.Track]:
|
|
"""
|
|
Removes duplicates from a list. Returns a list without duplicates.
|
|
"""
|
|
|
|
song_num = 0
|
|
|
|
while song_num < len(tracklist) - 1:
|
|
for index, song in enumerate(tracklist):
|
|
if (
|
|
tracklist[song_num].title == song.title
|
|
and tracklist[song_num].album == song.album
|
|
and tracklist[song_num].artists == song.artists
|
|
and index != song_num
|
|
):
|
|
tracklist.remove(song)
|
|
|
|
song_num += 1
|
|
|
|
return tracklist
|
|
|
|
|
|
def save_image(url: str, path: str) -> None:
|
|
"""
|
|
Saves an image from an url to a path.
|
|
"""
|
|
|
|
response = requests.get(url)
|
|
img = Image.open(BytesIO(response.content))
|
|
img.save(path, "JPEG")
|
|
|
|
|
|
def is_valid_file(filename: str) -> bool:
|
|
"""
|
|
Checks if a file is valid. Returns True if it is, False if it isn't.
|
|
"""
|
|
|
|
if filename.endswith(".flac") or filename.endswith(".mp3"):
|
|
return True
|
|
else:
|
|
return False
|
|
|
|
|
|
def create_config_dir() -> None:
|
|
"""
|
|
Creates the config directory if it doesn't exist.
|
|
"""
|
|
|
|
_home_dir = os.path.expanduser("~")
|
|
config_folder = os.path.join(_home_dir, app_dir)
|
|
|
|
dirs = ["", "images", "images/defaults", "images/artists", "images/thumbnails"]
|
|
|
|
for _dir in dirs:
|
|
path = os.path.join(config_folder, _dir)
|
|
|
|
try:
|
|
os.makedirs(path)
|
|
except FileExistsError:
|
|
pass
|
|
|
|
os.chmod(path, 0o755)
|
|
|
|
|
|
def get_all_songs() -> List[models.Track]:
|
|
"""
|
|
Gets all songs under the ~/ directory.
|
|
"""
|
|
print("Getting all songs...")
|
|
|
|
tracks: list[models.Track] = []
|
|
|
|
for track in instances.songs_instance.get_all_songs():
|
|
try:
|
|
os.chmod(os.path.join(track["filepath"]), 0o755)
|
|
except FileNotFoundError:
|
|
instances.songs_instance.remove_song_by_filepath(track["filepath"])
|
|
|
|
tracks.append(functions.create_track_class(track))
|
|
|
|
return tracks
|
|
|
|
|
|
def extract_colors(image) -> list:
|
|
colors = sorted(colorgram.extract(image, 2), key=lambda c: c.hsl.h)
|
|
|
|
formatted_colors = []
|
|
|
|
for color in colors:
|
|
color = f"rgb({color.rgb.r}, {color.rgb.g}, {color.rgb.b})"
|
|
formatted_colors.append(color)
|
|
|
|
return formatted_colors
|