|
import shutil |
|
from fastapi import FastAPI, HTTPException |
|
from deezspot.deezloader import DeeLogin |
|
import requests |
|
import os |
|
import logging |
|
from typing import Optional |
|
from fastapi.staticfiles import StaticFiles |
|
from dotenv import load_dotenv |
|
|
|
|
|
logging.basicConfig(level=logging.INFO) |
|
logger = logging.getLogger(__name__) |
|
|
|
app = FastAPI(title="Deezer API") |
|
|
|
load_dotenv() |
|
|
|
|
|
os.makedirs("downloads", exist_ok=True) |
|
app.mount("/downloads", StaticFiles(directory="downloads"), name="downloads") |
|
|
|
|
|
DEEZER_API_URL = "https://api.deezer.com" |
|
|
|
|
|
ARL_TOKEN = os.getenv('ARL') |
|
dl = DeeLogin(arl=ARL_TOKEN) |
|
|
|
@app.get("/") |
|
def read_root(): |
|
return {"message": "running"} |
|
|
|
|
|
def get_track_info(track_id: str): |
|
try: |
|
response = requests.get(f"{DEEZER_API_URL}/track/{track_id}") |
|
if response.status_code != 200: |
|
raise HTTPException(status_code=404, detail="Track not found") |
|
return response.json() |
|
except requests.exceptions.RequestException as e: |
|
logger.error(f"Network error fetching track metadata: {e}") |
|
raise HTTPException(status_code=500, detail=str(e)) |
|
except Exception as e: |
|
logger.error(f"Error fetching track metadata: {e}") |
|
raise HTTPException(status_code=500, detail=str(e)) |
|
|
|
|
|
@app.get("/track/{track_id}") |
|
def get_track(track_id: str): |
|
return get_track_info(track_id) |
|
|
|
|
|
@app.post("/download/track/{track_id}") |
|
def download_track(track_id: str, quality: str = "MP3_320"): |
|
try: |
|
|
|
track_info = get_track_info(track_id) |
|
track_link = track_info.get("link") |
|
if not track_link: |
|
raise HTTPException(status_code=404, detail="Track link not found") |
|
|
|
|
|
track_title = track_info.get("title", "track") |
|
artist_name = track_info.get("artist", {}).get("name", "unknown") |
|
expected_filename = f"{artist_name} - {track_title}.mp3".replace("/", "_") |
|
|
|
|
|
for root, dirs, files in os.walk("downloads"): |
|
for file in files: |
|
os.remove(os.path.join(root, file)) |
|
for dir in dirs: |
|
shutil.rmtree(os.path.join(root, dir)) |
|
|
|
|
|
logger.info(f"Downloading track: {expected_filename}") |
|
dl.download_trackdee( |
|
link_track=track_link, |
|
output_dir="downloads", |
|
quality_download=quality, |
|
recursive_quality=False, |
|
recursive_download=False |
|
) |
|
|
|
|
|
mp3_filepath = None |
|
for root, dirs, files in os.walk("downloads"): |
|
for file in files: |
|
if file.endswith('.mp3'): |
|
mp3_filepath = os.path.join(root, file) |
|
break |
|
if mp3_filepath: |
|
break |
|
|
|
if not mp3_filepath: |
|
raise HTTPException(status_code=500, detail="MP3 file not found after download") |
|
|
|
|
|
relative_path = os.path.relpath(mp3_filepath, "downloads") |
|
download_url = f"/downloads/{relative_path}" |
|
logger.info(f"Download successful: {download_url}") |
|
return {"download_url": download_url} |
|
except Exception as e: |
|
logger.error(f"Error downloading track: {e}") |
|
raise HTTPException(status_code=500, detail=str(e)) |
|
|
|
|
|
@app.get("/search") |
|
def search_tracks(query: str, limit: Optional[int] = 10): |
|
try: |
|
response = requests.get(f"{DEEZER_API_URL}/search", params={"q": query, "limit": limit}) |
|
return response.json() |
|
except requests.exceptions.RequestException as e: |
|
logger.error(f"Network error searching tracks: {e}") |
|
raise HTTPException(status_code=500, detail=str(e)) |
|
except Exception as e: |
|
logger.error(f"Error searching tracks: {e}") |
|
raise HTTPException(status_code=500, detail=str(e)) |