Added Radarr support
This commit is contained in:
parent
7b86003d9f
commit
a9ec3e3ebc
11
.env.example
11
.env.example
@ -4,9 +4,20 @@ ANILIST_USERNAME=
|
|||||||
# The Sonarr instance to sync to
|
# The Sonarr instance to sync to
|
||||||
SONARR_API_URL=
|
SONARR_API_URL=
|
||||||
SONARR_API_KEY=
|
SONARR_API_KEY=
|
||||||
|
|
||||||
# Root folder to save the shows to
|
# Root folder to save the shows to
|
||||||
SONARR_FOLDER_PATH=
|
SONARR_FOLDER_PATH=
|
||||||
|
|
||||||
# IDs of the quality and language profiles to use
|
# IDs of the quality and language profiles to use
|
||||||
SONARR_QUALITY_PROFILE=4
|
SONARR_QUALITY_PROFILE=4
|
||||||
SONARR_LANGUAGE_PROFILE=1
|
SONARR_LANGUAGE_PROFILE=1
|
||||||
|
|
||||||
|
# The Radarr instance to sync to
|
||||||
|
RADARR_API_URL=
|
||||||
|
RADARR_API_KEY=
|
||||||
|
|
||||||
|
# Root folder to save the movies to
|
||||||
|
RADARR_FOLDER_PATH=
|
||||||
|
|
||||||
|
# ID of the quality profile to use
|
||||||
|
RADARR_QUALITY_PROFILE=4
|
||||||
|
@ -1,35 +1,43 @@
|
|||||||
import os
|
import os
|
||||||
import typing
|
|
||||||
from dotenv import load_dotenv
|
from dotenv import load_dotenv
|
||||||
from al_arr_sync.anilist import AniListClient
|
from al_arr_sync.anilist import AniListClient
|
||||||
from al_arr_sync.sonarr import SonarrClient
|
from al_arr_sync.sonarr import SonarrClient
|
||||||
from al_arr_sync.types import AnyDict
|
from al_arr_sync.radarr import RadarrClient
|
||||||
|
from al_arr_sync.types import DlAutomator
|
||||||
|
|
||||||
load_dotenv()
|
load_dotenv()
|
||||||
|
|
||||||
|
|
||||||
def main() -> int:
|
def main() -> int:
|
||||||
al = AniListClient()
|
al = AniListClient()
|
||||||
sonarr = SonarrClient.from_env()
|
sonarr: DlAutomator = SonarrClient.from_env()
|
||||||
|
radarr: DlAutomator = RadarrClient.from_env()
|
||||||
|
|
||||||
username = os.environ["ANILIST_USERNAME"]
|
username = os.environ["ANILIST_USERNAME"]
|
||||||
media = al.currently_watching(username)
|
media = al.currently_watching(username)
|
||||||
|
|
||||||
series: typing.List[AnyDict] = []
|
|
||||||
for entry in media:
|
for entry in media:
|
||||||
media_format = entry["media"]["format"]
|
media_format = entry["media"]["format"]
|
||||||
if media_format == "TV":
|
show_name = entry["media"]["title"]["english"]
|
||||||
series.append(entry)
|
|
||||||
|
client: DlAutomator
|
||||||
|
if media_format == "TV":
|
||||||
|
client = sonarr
|
||||||
|
elif media_format == "MOVIE":
|
||||||
|
client = radarr
|
||||||
|
else:
|
||||||
|
continue
|
||||||
|
|
||||||
|
results = client.lookup_series(show_name)
|
||||||
|
if len(results) == 0:
|
||||||
|
print(f"No results found for: {show_name}")
|
||||||
|
continue
|
||||||
|
|
||||||
for show in series:
|
|
||||||
show_name = show["media"]["title"]["english"]
|
|
||||||
results = sonarr.lookup_series(show_name)
|
|
||||||
try:
|
try:
|
||||||
sonarr.add_series(results[0])
|
client.add_series(results[0])
|
||||||
print(f"Successfully added series {show_name}")
|
print(f"Successfully added: {show_name}")
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
print(e)
|
print(e)
|
||||||
continue
|
|
||||||
|
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
|
64
al_arr_sync/radarr.py
Normal file
64
al_arr_sync/radarr.py
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
import typing
|
||||||
|
from requests import Session
|
||||||
|
from requests import PreparedRequest
|
||||||
|
import os
|
||||||
|
from urllib.parse import urljoin
|
||||||
|
from al_arr_sync.types import AnyDict
|
||||||
|
|
||||||
|
|
||||||
|
class RadarrClient:
|
||||||
|
def __init__(self, radarr_url: str, api_key: str) -> None:
|
||||||
|
self.radarr_url = radarr_url
|
||||||
|
self.api_key = api_key
|
||||||
|
self.http_session = Session()
|
||||||
|
|
||||||
|
@staticmethod
|
||||||
|
def from_env() -> "RadarrClient":
|
||||||
|
return RadarrClient(
|
||||||
|
radarr_url=os.environ["RADARR_API_URL"],
|
||||||
|
api_key=os.environ["RADARR_API_KEY"],
|
||||||
|
)
|
||||||
|
|
||||||
|
def _prepare_request(
|
||||||
|
self,
|
||||||
|
endpoint: str,
|
||||||
|
method: str = "GET",
|
||||||
|
params: AnyDict = {},
|
||||||
|
json: typing.Optional[AnyDict] = None,
|
||||||
|
) -> PreparedRequest:
|
||||||
|
url = urljoin(self.radarr_url, endpoint)
|
||||||
|
headers = {"X-Api-Key": self.api_key}
|
||||||
|
|
||||||
|
req = PreparedRequest()
|
||||||
|
req.prepare(method=method, url=url, headers=headers, params=params, json=json)
|
||||||
|
return req
|
||||||
|
|
||||||
|
def lookup_series(self, query: str) -> typing.List[AnyDict]:
|
||||||
|
req = self._prepare_request("/api/v3/movie/lookup", params={"term": query})
|
||||||
|
resp = self.http_session.send(req)
|
||||||
|
return resp.json()
|
||||||
|
|
||||||
|
def add_series(self, *series: AnyDict):
|
||||||
|
for show in series:
|
||||||
|
payload: AnyDict = show.copy()
|
||||||
|
payload.update(
|
||||||
|
{
|
||||||
|
"addOptions": {
|
||||||
|
"searchForMovie": True,
|
||||||
|
"ignoreEpisodesWithFiles": False,
|
||||||
|
"ignoreEpisodesWithoutFiles": False,
|
||||||
|
},
|
||||||
|
"rootFolderPath": os.environ["RADARR_FOLDER_PATH"],
|
||||||
|
"qualityProfileId": int(os.environ["RADARR_QUALITY_PROFILE"]),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
req = self._prepare_request("/api/v3/movie", method="POST", json=payload)
|
||||||
|
resp = self.http_session.send(req)
|
||||||
|
|
||||||
|
if resp.status_code != 201:
|
||||||
|
resp_data = resp.json()
|
||||||
|
error_codes = map(lambda x: x["errorCode"], resp_data)
|
||||||
|
if "SeriesExistsValidator" in error_codes:
|
||||||
|
raise Exception(f"Movie already exists: {show['title']}")
|
||||||
|
|
||||||
|
raise Exception(f"Failed to add movie: {show['title']}:\n{resp_data}")
|
@ -33,8 +33,8 @@ class SonarrClient:
|
|||||||
req.prepare(method=method, url=url, headers=headers, params=params, json=json)
|
req.prepare(method=method, url=url, headers=headers, params=params, json=json)
|
||||||
return req
|
return req
|
||||||
|
|
||||||
def lookup_series(self, title: str) -> typing.List[AnyDict]:
|
def lookup_series(self, query: str) -> typing.List[AnyDict]:
|
||||||
req = self._prepare_request("/api/v3/series/lookup", params={"term": title})
|
req = self._prepare_request("/api/v3/series/lookup", params={"term": query})
|
||||||
resp = self.http_session.send(req)
|
resp = self.http_session.send(req)
|
||||||
return resp.json()
|
return resp.json()
|
||||||
|
|
||||||
@ -57,4 +57,9 @@ class SonarrClient:
|
|||||||
resp = self.http_session.send(req)
|
resp = self.http_session.send(req)
|
||||||
|
|
||||||
if resp.status_code != 201:
|
if resp.status_code != 201:
|
||||||
raise Exception(f"Failed to add series {show['title']}:\n{resp.json()}")
|
resp_data = resp.json()
|
||||||
|
error_codes = map(lambda x: x["errorCode"], resp_data)
|
||||||
|
if "SeriesExistsValidator" in error_codes:
|
||||||
|
raise Exception(f"Series already exists: {show['title']}")
|
||||||
|
|
||||||
|
raise Exception(f"Failed to add series: {show['title']}:\n{resp_data}")
|
||||||
|
@ -1,3 +1,11 @@
|
|||||||
import typing
|
import typing
|
||||||
|
|
||||||
AnyDict = typing.Dict[typing.Any, typing.Any]
|
AnyDict = typing.Dict[typing.Any, typing.Any]
|
||||||
|
|
||||||
|
|
||||||
|
class DlAutomator(typing.Protocol):
|
||||||
|
def lookup_series(self, query: str) -> typing.List[AnyDict]:
|
||||||
|
...
|
||||||
|
|
||||||
|
def add_series(self, *series: AnyDict):
|
||||||
|
...
|
||||||
|
Loading…
x
Reference in New Issue
Block a user