Initial import: Music_Server, MusicFree, catalog-sync
This commit is contained in:
@@ -0,0 +1,119 @@
|
||||
'''
|
||||
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
|
||||
Reference in New Issue
Block a user