119 lines
7.5 KiB
Python
119 lines
7.5 KiB
Python
'''
|
|
Function:
|
|
Implementation of MP3JuiceMusicClient: https://mp3juice.co/
|
|
Author:
|
|
Zhenchao Jin
|
|
WeChat Official Account (微信公众号):
|
|
Charles的皮卡丘
|
|
'''
|
|
import re
|
|
import copy
|
|
import time
|
|
import base64
|
|
import json_repair
|
|
from urllib.parse import quote
|
|
from itertools import zip_longest
|
|
from urllib.parse import urlencode
|
|
from rich.progress import Progress
|
|
from ..sources import BaseMusicClient
|
|
from ..utils import legalizestring, usesearchheaderscookies, resp2json, byte2mb, SongInfo
|
|
|
|
|
|
'''MP3JuiceMusicClient'''
|
|
class MP3JuiceMusicClient(BaseMusicClient):
|
|
source = 'MP3JuiceMusicClient'
|
|
def __init__(self, **kwargs):
|
|
kwargs['search_size_per_source'] = kwargs['search_size_per_source'] * 2
|
|
super(MP3JuiceMusicClient, self).__init__(**kwargs)
|
|
self.default_search_headers = {"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36", "Referer": "https://mp3juice.sc/", "Origin": "https://mp3juice.sc"}
|
|
self.default_download_headers = {"User-Agent": "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36", "Referer": "https://mp3juice.sc/", "Origin": "https://mp3juice.sc"}
|
|
self.default_headers = self.default_search_headers
|
|
self._initsession()
|
|
'''_getdynamicconfig'''
|
|
def _getdynamicconfig(self, request_overrides: dict = None):
|
|
request_overrides = request_overrides or {}
|
|
(resp := self.get(f"https://mp3juice.as/?t={int(time.time() * 1000)}", **request_overrides)).raise_for_status()
|
|
match = re.search(r"var\s+json\s*=\s*JSON\.parse\('(.+?)'\);", resp.text)
|
|
if not match: match = re.search(r"var\s+json\s*=\s*(\[.+?\]);", resp.text)
|
|
return json_repair.loads(match.group(1))
|
|
'''_calculateauth'''
|
|
def _calculateauth(self, raw_data):
|
|
data_arr, should_reverse, offset_arr, result_chars = raw_data[0], raw_data[1], raw_data[2], []; offset_len = len(offset_arr)
|
|
for t in range(len(data_arr)): result_chars.append(chr(data_arr[t] - offset_arr[offset_len - (t + 1)]))
|
|
if should_reverse: result_chars.reverse()
|
|
full_token = "".join(result_chars)
|
|
return full_token[:32]
|
|
'''_constructsearchurls'''
|
|
def _constructsearchurls(self, keyword: str, rule: dict = None, request_overrides: dict = None):
|
|
# init
|
|
rule, request_overrides = rule or {}, request_overrides or {}
|
|
config = self._getdynamicconfig(); auth_token = self._calculateauth(config)
|
|
# search rules
|
|
default_rule = {'k': auth_token, 'y': 's', 'q': base64.b64encode(quote(keyword, safe="").encode("utf-8")).decode("utf-8"), 't': str(int(time.time()))}
|
|
default_rule.update(rule)
|
|
# construct search urls based on search rules
|
|
base_url = 'https://mp3juice.sc/api/v1/search?'
|
|
page_rule = copy.deepcopy(default_rule)
|
|
search_urls = [{'url': base_url + urlencode(page_rule), 'auth_token': auth_token, 'param_key': chr(config[6])}]
|
|
self.search_size_per_page = self.search_size_per_source
|
|
# return
|
|
return search_urls
|
|
'''_search'''
|
|
@usesearchheaderscookies
|
|
def _search(self, keyword: str = '', search_url: dict = None, request_overrides: dict = None, song_infos: list = [], progress: Progress = None, progress_id: int = 0):
|
|
# init
|
|
request_overrides, search_meta = request_overrides or {}, copy.deepcopy(search_url)
|
|
search_url, auth_token, param_key = search_meta['url'], search_meta['auth_token'], search_meta['param_key']
|
|
# successful
|
|
try:
|
|
# --search results
|
|
(resp := self.get(search_url, allow_redirects=True, **request_overrides)).raise_for_status()
|
|
search_results_yt, search_results_sc = [], []
|
|
for item in resp2json(resp)["yt"]: item['root_source'] = 'YouTube'; search_results_yt.append(item)
|
|
for item in resp2json(resp)["sc"]: item['root_source'] = 'SoundCloud'; search_results_sc.append(item)
|
|
search_results = [x for ab in zip_longest(search_results_yt, search_results_sc) for x in ab if x is not None]
|
|
for search_result in search_results:
|
|
# --judgement for search_size
|
|
if self.strict_limit_search_size_per_page and len(song_infos) >= self.search_size_per_page: break
|
|
# --download results
|
|
if not isinstance(search_result, dict) or ('id' not in search_result): continue
|
|
if search_result['root_source'] in ['SoundCloud'] and ('id_base64' not in search_result or 'title_base64' not in search_result): continue
|
|
song_info, download_result = SongInfo(source=self.source, root_source=search_result['root_source']), dict()
|
|
# ----SoundCloud
|
|
if search_result['root_source'] in ['SoundCloud']:
|
|
download_url = f"https://thetacloud.org/s/{search_result['id_base64']}/{search_result['title_base64']}/"
|
|
# ----YouTube
|
|
else:
|
|
params = {param_key: auth_token, 't': str(int(time.time()))}
|
|
try: (init_resp := self.get('https://theta.thetacloud.org/api/v1/init?', params=params, **request_overrides)).raise_for_status()
|
|
except Exception: continue
|
|
download_result['init'] = resp2json(resp=init_resp)
|
|
if not (convert_url := download_result['init'].get('convertURL', '')): continue
|
|
convert_url = f'{convert_url}&v={search_result["id"]}&f=mp3&t={str(int(time.time()))}'
|
|
try: (convert_resp := self.get(convert_url, **request_overrides)).raise_for_status()
|
|
except Exception: continue
|
|
download_result['convert'] = resp2json(resp=convert_resp)
|
|
if not (redirect_url := download_result['convert'].get('redirectURL', '')): continue
|
|
try: (resp := self.get(redirect_url, **request_overrides)).raise_for_status()
|
|
except Exception: continue
|
|
download_result['redirect'] = resp2json(resp=resp)
|
|
if not (download_url := download_result['redirect'].get('downloadURL', '')): continue
|
|
song_info = SongInfo(
|
|
raw_data={'search': search_result, 'download': download_result, 'lyric': {}}, source=self.source, song_name=legalizestring(search_result.get('title')), singers='NULL',
|
|
album='NULL', ext='mp3', file_size_bytes=None, file_size=None, identifier=search_result['id'], duration='-:-:-', lyric='NULL', cover_url=None, download_url=download_url,
|
|
download_url_status=self.audio_link_tester.test(download_url, request_overrides), root_source=search_result['root_source'],
|
|
)
|
|
if not song_info.with_valid_download_url: continue
|
|
# ----you have to download the music contents immediately, otherwise the links will fail.
|
|
song_info.downloaded_contents = self.get(download_url, **request_overrides).content
|
|
song_info.file_size_bytes = song_info.downloaded_contents.__sizeof__()
|
|
song_info.file_size = byte2mb(song_info.file_size_bytes)
|
|
# --append to song_infos
|
|
song_infos.append(song_info)
|
|
# --update progress
|
|
progress.update(progress_id, description=f"{self.source}.search >>> {search_url} (Success)")
|
|
# failure
|
|
except Exception as err:
|
|
progress.update(progress_id, description=f"{self.source}.search >>> {search_url} (Error: {err})")
|
|
# return
|
|
return song_infos |