Compare commits
4 Commits
e2b7c5475f
...
ae921c3626
Author | SHA1 | Date | |
---|---|---|---|
ae921c3626 | |||
f52b5e6325 | |||
25eceb727b | |||
e77afa584b |
@ -6,8 +6,8 @@ logging.getLogger().setLevel(logging.DEBUG)
|
|||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
commands = [
|
commands = [
|
||||||
"s: #a Ruffiction",
|
"s: #a Crystal F",
|
||||||
"d: 8",
|
"d: 20",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import json
|
import json
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass, field
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
from typing import List, Optional
|
from typing import List, Optional
|
||||||
from functools import lru_cache
|
from functools import lru_cache
|
||||||
@ -18,6 +18,8 @@ class CacheAttribute:
|
|||||||
created: datetime
|
created: datetime
|
||||||
expires: datetime
|
expires: datetime
|
||||||
|
|
||||||
|
additional_info: dict = field(default_factory=dict)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def id(self):
|
def id(self):
|
||||||
return f"{self.module}_{self.name}"
|
return f"{self.module}_{self.name}"
|
||||||
@ -32,6 +34,12 @@ class CacheAttribute:
|
|||||||
return self.__dict__ == other.__dict__
|
return self.__dict__ == other.__dict__
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class CacheResult:
|
||||||
|
content: bytes
|
||||||
|
attribute: CacheAttribute
|
||||||
|
|
||||||
|
|
||||||
class Cache:
|
class Cache:
|
||||||
def __init__(self, module: str, logger: logging.Logger):
|
def __init__(self, module: str, logger: logging.Logger):
|
||||||
self.module = module
|
self.module = module
|
||||||
@ -49,6 +57,7 @@ class Cache:
|
|||||||
|
|
||||||
self._time_fields = {"created", "expires"}
|
self._time_fields = {"created", "expires"}
|
||||||
with self.index.open("r") as i:
|
with self.index.open("r") as i:
|
||||||
|
try:
|
||||||
for c in json.loads(i.read()):
|
for c in json.loads(i.read()):
|
||||||
for key in self._time_fields:
|
for key in self._time_fields:
|
||||||
c[key] = datetime.fromisoformat(c[key])
|
c[key] = datetime.fromisoformat(c[key])
|
||||||
@ -56,6 +65,8 @@ class Cache:
|
|||||||
ca = CacheAttribute(**c)
|
ca = CacheAttribute(**c)
|
||||||
self.cached_attributes.append(ca)
|
self.cached_attributes.append(ca)
|
||||||
self._id_to_attribute[ca.id] = ca
|
self._id_to_attribute[ca.id] = ca
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
pass
|
||||||
|
|
||||||
@lru_cache()
|
@lru_cache()
|
||||||
def _init_module(self, module: str) -> Path:
|
def _init_module(self, module: str) -> Path:
|
||||||
@ -100,7 +111,7 @@ class Cache:
|
|||||||
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
def set(self, content: bytes, name: str, expires_in: float = 10, module: str = ""):
|
def set(self, content: bytes, name: str, expires_in: float = 10, module: str = "", additional_info: dict = None):
|
||||||
"""
|
"""
|
||||||
:param content:
|
:param content:
|
||||||
:param module:
|
:param module:
|
||||||
@ -111,6 +122,7 @@ class Cache:
|
|||||||
if name == "":
|
if name == "":
|
||||||
return
|
return
|
||||||
|
|
||||||
|
additional_info = additional_info or {}
|
||||||
module = self.module if module == "" else module
|
module = self.module if module == "" else module
|
||||||
|
|
||||||
module_path = self._init_module(module)
|
module_path = self._init_module(module)
|
||||||
@ -120,6 +132,7 @@ class Cache:
|
|||||||
name=name,
|
name=name,
|
||||||
created=datetime.now(),
|
created=datetime.now(),
|
||||||
expires=datetime.now() + timedelta(days=expires_in),
|
expires=datetime.now() + timedelta(days=expires_in),
|
||||||
|
additional_info=additional_info,
|
||||||
)
|
)
|
||||||
self._write_attribute(cache_attribute)
|
self._write_attribute(cache_attribute)
|
||||||
|
|
||||||
@ -128,7 +141,7 @@ class Cache:
|
|||||||
self.logger.debug(f"writing cache to {cache_path}")
|
self.logger.debug(f"writing cache to {cache_path}")
|
||||||
content_file.write(content)
|
content_file.write(content)
|
||||||
|
|
||||||
def get(self, name: str) -> Optional[bytes]:
|
def get(self, name: str) -> Optional[CacheResult]:
|
||||||
path = fit_to_file_system(Path(self._dir, self.module, name), hidden_ok=True)
|
path = fit_to_file_system(Path(self._dir, self.module, name), hidden_ok=True)
|
||||||
|
|
||||||
if not path.is_file():
|
if not path.is_file():
|
||||||
@ -140,7 +153,7 @@ class Cache:
|
|||||||
return
|
return
|
||||||
|
|
||||||
with path.open("rb") as f:
|
with path.open("rb") as f:
|
||||||
return f.read()
|
return CacheResult(content=f.read(), attribute=existing_attribute)
|
||||||
|
|
||||||
def clean(self):
|
def clean(self):
|
||||||
keep = set()
|
keep = set()
|
||||||
|
@ -125,12 +125,17 @@ class Connection:
|
|||||||
|
|
||||||
return headers
|
return headers
|
||||||
|
|
||||||
def save(self, r: requests.Response, name: str, error: bool = False, **kwargs):
|
def save(self, r: requests.Response, name: str, error: bool = False, no_update_if_valid_exists: bool = False, **kwargs):
|
||||||
n_kwargs = {}
|
n_kwargs = {}
|
||||||
if error:
|
if error:
|
||||||
n_kwargs["module"] = "failed_requests"
|
n_kwargs["module"] = "failed_requests"
|
||||||
|
|
||||||
self.cache.set(r.content, name, expires_in=kwargs.get("expires_in", self.cache_expiring_duration), **n_kwargs)
|
if self.cache.get(name) is not None and no_update_if_valid_exists:
|
||||||
|
return
|
||||||
|
|
||||||
|
self.cache.set(r.content, name, expires_in=kwargs.get("expires_in", self.cache_expiring_duration), additional_info={
|
||||||
|
"encoding": r.encoding,
|
||||||
|
}, **n_kwargs)
|
||||||
|
|
||||||
def request(
|
def request(
|
||||||
self,
|
self,
|
||||||
@ -145,6 +150,7 @@ class Connection:
|
|||||||
sleep_after_404: float = None,
|
sleep_after_404: float = None,
|
||||||
is_heartbeat: bool = False,
|
is_heartbeat: bool = False,
|
||||||
disable_cache: bool = None,
|
disable_cache: bool = None,
|
||||||
|
enable_cache_readonly: bool = False,
|
||||||
method: str = None,
|
method: str = None,
|
||||||
name: str = "",
|
name: str = "",
|
||||||
exclude_headers: List[str] = None,
|
exclude_headers: List[str] = None,
|
||||||
@ -178,17 +184,23 @@ class Connection:
|
|||||||
|
|
||||||
request_url = parsed_url.geturl() if not raw_url else url
|
request_url = parsed_url.geturl() if not raw_url else url
|
||||||
|
|
||||||
if name != "" and not disable_cache:
|
if name != "" and (not disable_cache or enable_cache_readonly):
|
||||||
cached = self.cache.get(name)
|
cached = self.cache.get(name)
|
||||||
|
|
||||||
if cached is not None:
|
if cached is not None:
|
||||||
request_trace(f"{trace_string}\t[cached]")
|
request_trace(f"{trace_string}\t[cached]")
|
||||||
|
|
||||||
with responses.RequestsMock() as resp:
|
with responses.RequestsMock() as resp:
|
||||||
|
additional_info = cached.attribute.additional_info
|
||||||
|
|
||||||
|
body = cached.content
|
||||||
|
if "encoding" in additional_info:
|
||||||
|
body = body.decode(additional_info["encoding"])
|
||||||
|
|
||||||
resp.add(
|
resp.add(
|
||||||
method=method,
|
method=method,
|
||||||
url=request_url,
|
url=request_url,
|
||||||
body=cached,
|
body=body,
|
||||||
)
|
)
|
||||||
return requests.request(method=method, url=url, timeout=timeout, headers=headers, **kwargs)
|
return requests.request(method=method, url=url, timeout=timeout, headers=headers, **kwargs)
|
||||||
|
|
||||||
|
@ -451,7 +451,7 @@ class Page:
|
|||||||
source = sources[0]
|
source = sources[0]
|
||||||
|
|
||||||
if not found_on_disc:
|
if not found_on_disc:
|
||||||
r = self.download_song_to_target(source=source, target=temp_target, desc=song.title)
|
r = self.download_song_to_target(source=source, target=temp_target, desc=song.option_string)
|
||||||
|
|
||||||
if not r.is_fatal_error:
|
if not r.is_fatal_error:
|
||||||
r.merge(self._post_process_targets(song, temp_target,
|
r.merge(self._post_process_targets(song, temp_target,
|
||||||
|
@ -1128,4 +1128,4 @@ class Musify(Page):
|
|||||||
|
|
||||||
self.LOGGER.warning(f"The source has no audio link. Falling back to {endpoint}.")
|
self.LOGGER.warning(f"The source has no audio link. Falling back to {endpoint}.")
|
||||||
|
|
||||||
return self.stream_connection.stream_into(endpoint, target, raw_url=True, exclude_headers=["Host"])
|
return self.stream_connection.stream_into(endpoint, target, raw_url=True, exclude_headers=["Host"], name=desc)
|
||||||
|
@ -2,6 +2,7 @@ from typing import List, Optional
|
|||||||
from enum import Enum
|
from enum import Enum
|
||||||
|
|
||||||
from ...utils.config import youtube_settings, logging_settings
|
from ...utils.config import youtube_settings, logging_settings
|
||||||
|
from ...utils.string_processing import clean_song_title
|
||||||
from ...objects import Source, DatabaseObject
|
from ...objects import Source, DatabaseObject
|
||||||
from ..abstract import Page
|
from ..abstract import Page
|
||||||
from ...objects import (
|
from ...objects import (
|
||||||
@ -59,7 +60,7 @@ def parse_run_element(run_element: dict) -> Optional[DatabaseObject]:
|
|||||||
|
|
||||||
if element_type == PageType.SONG or (element_type == PageType.VIDEO and not youtube_settings["youtube_music_clean_data"]) or (element_type == PageType.OFFICIAL_MUSIC_VIDEO and not youtube_settings["youtube_music_clean_data"]):
|
if element_type == PageType.SONG or (element_type == PageType.VIDEO and not youtube_settings["youtube_music_clean_data"]) or (element_type == PageType.OFFICIAL_MUSIC_VIDEO and not youtube_settings["youtube_music_clean_data"]):
|
||||||
source = Source(SOURCE_PAGE, f"https://music.youtube.com/watch?v={element_id}")
|
source = Source(SOURCE_PAGE, f"https://music.youtube.com/watch?v={element_id}")
|
||||||
return Song(title=element_text, source_list=[source])
|
return Song(title=clean_song_title(element_text), source_list=[source])
|
||||||
|
|
||||||
if element_type == PageType.ARTIST or (element_type == PageType.CHANNEL and not youtube_settings["youtube_music_clean_data"]):
|
if element_type == PageType.ARTIST or (element_type == PageType.CHANNEL and not youtube_settings["youtube_music_clean_data"]):
|
||||||
source = Source(SOURCE_PAGE, f"https://music.youtube.com/channel/{element_id}")
|
source = Source(SOURCE_PAGE, f"https://music.youtube.com/channel/{element_id}")
|
||||||
|
@ -171,7 +171,7 @@ class YoutubeMusic(SuperYouTube):
|
|||||||
def __init__(self, *args, ydl_opts: dict = None, **kwargs):
|
def __init__(self, *args, ydl_opts: dict = None, **kwargs):
|
||||||
self.yt_music_connection: YoutubeMusicConnection = YoutubeMusicConnection(
|
self.yt_music_connection: YoutubeMusicConnection = YoutubeMusicConnection(
|
||||||
logger=self.LOGGER,
|
logger=self.LOGGER,
|
||||||
accept_language="en-US,en;q=0.5"
|
accept_language="en-US,en;q=0.5",
|
||||||
)
|
)
|
||||||
self.credentials: YouTubeMusicCredentials = YouTubeMusicCredentials(
|
self.credentials: YouTubeMusicCredentials = YouTubeMusicCredentials(
|
||||||
api_key=youtube_settings["youtube_music_api_key"],
|
api_key=youtube_settings["youtube_music_api_key"],
|
||||||
@ -212,7 +212,7 @@ class YoutubeMusic(SuperYouTube):
|
|||||||
search for: "innertubeApiKey"
|
search for: "innertubeApiKey"
|
||||||
"""
|
"""
|
||||||
|
|
||||||
r = self.yt_music_connection.get("https://music.youtube.com/")
|
r = self.yt_music_connection.get("https://music.youtube.com/", name="youtube_music_index.html", disable_cache=True, enable_cache_readonly=True)
|
||||||
if r is None:
|
if r is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -232,7 +232,7 @@ class YoutubeMusic(SuperYouTube):
|
|||||||
'set_ytc': 'true',
|
'set_ytc': 'true',
|
||||||
'set_apyt': 'true',
|
'set_apyt': 'true',
|
||||||
'set_eom': 'false'
|
'set_eom': 'false'
|
||||||
})
|
}, disable_cache=True)
|
||||||
if r is None:
|
if r is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -247,9 +247,9 @@ class YoutubeMusic(SuperYouTube):
|
|||||||
# save cookies in settings
|
# save cookies in settings
|
||||||
youtube_settings["youtube_music_consent_cookies"] = cookie_dict
|
youtube_settings["youtube_music_consent_cookies"] = cookie_dict
|
||||||
else:
|
else:
|
||||||
self.yt_music_connection.save(r, "index.html")
|
self.yt_music_connection.save(r, "youtube_music_index.html", no_update_if_valid_exists=True)
|
||||||
|
|
||||||
r = self.yt_music_connection.get("https://music.youtube.com/", name="index.html")
|
r = self.yt_music_connection.get("https://music.youtube.com/", name="youtube_music_index.html")
|
||||||
if r is None:
|
if r is None:
|
||||||
return
|
return
|
||||||
|
|
||||||
@ -374,7 +374,8 @@ class YoutubeMusic(SuperYouTube):
|
|||||||
},
|
},
|
||||||
headers={
|
headers={
|
||||||
"Referer": get_youtube_url(path=f"/search", query=f"q={urlescaped_query}")
|
"Referer": get_youtube_url(path=f"/search", query=f"q={urlescaped_query}")
|
||||||
}
|
},
|
||||||
|
name=f"search_{search_query}.json"
|
||||||
)
|
)
|
||||||
|
|
||||||
if r is None:
|
if r is None:
|
||||||
@ -411,7 +412,8 @@ class YoutubeMusic(SuperYouTube):
|
|||||||
json={
|
json={
|
||||||
"browseId": browse_id,
|
"browseId": browse_id,
|
||||||
"context": {**self.credentials.context, "adSignalsInfo": {"params": []}}
|
"context": {**self.credentials.context, "adSignalsInfo": {"params": []}}
|
||||||
}
|
},
|
||||||
|
name=f"fetch_artist_{browse_id}.json"
|
||||||
)
|
)
|
||||||
if r is None:
|
if r is None:
|
||||||
return artist
|
return artist
|
||||||
@ -454,7 +456,8 @@ class YoutubeMusic(SuperYouTube):
|
|||||||
json={
|
json={
|
||||||
"browseId": browse_id,
|
"browseId": browse_id,
|
||||||
"context": {**self.credentials.context, "adSignalsInfo": {"params": []}}
|
"context": {**self.credentials.context, "adSignalsInfo": {"params": []}}
|
||||||
}
|
},
|
||||||
|
name=f"fetch_album_{browse_id}.json"
|
||||||
)
|
)
|
||||||
if r is None:
|
if r is None:
|
||||||
return album
|
return album
|
||||||
|
Loading…
Reference in New Issue
Block a user