Merge pull request #116 from JazzFisch/additional-widgets

Add Readarr and SABnzbd widgets
This commit is contained in:
Ben Phelps 2022-09-12 10:23:49 +03:00 committed by GitHub
commit ee79335eff
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
17 changed files with 197 additions and 0 deletions

View file

@ -19,6 +19,7 @@
* Service Integration * Service Integration
- Currently supports Sonarr, Radarr, Ombi, Emby, Jellyfin, Tautulli (Plex), Overseerr, Jellyseerr ([ilusi0n](https://github.com/benphelps/homepage/pull/34)), NZBGet, ruTorrent - Currently supports Sonarr, Radarr, Ombi, Emby, Jellyfin, Tautulli (Plex), Overseerr, Jellyseerr ([ilusi0n](https://github.com/benphelps/homepage/pull/34)), NZBGet, ruTorrent
- Portainer, Traefik, Speedtest Tracker, PiHole, Nginx Proxy Manager ([aidenpwnz](https://github.com/benphelps/homepage/pull/45)) - Portainer, Traefik, Speedtest Tracker, PiHole, Nginx Proxy Manager ([aidenpwnz](https://github.com/benphelps/homepage/pull/45))
- Readarr, SABnzbd ([JazzFisch]())
* Information & Utility Widgets * Information & Utility Widgets
- System Stats (Disk, CPU, Memory) - System Stats (Disk, CPU, Memory)
- Weather via WeatherAPI.com or OpenWeatherMap ([AlexFullmoon](https://github.com/benphelps/homepage/pull/25)) - Weather via WeatherAPI.com or OpenWeatherMap ([AlexFullmoon](https://github.com/benphelps/homepage/pull/25))

View file

@ -36,6 +36,11 @@
"remaining": "Verblieben", "remaining": "Verblieben",
"downloaded": "Heruntergeladen" "downloaded": "Heruntergeladen"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Aktiv", "active": "Aktiv",
"upload": "Hochladen", "upload": "Hochladen",
@ -51,6 +56,11 @@
"queued": "In Warteschlange", "queued": "In Warteschlange",
"movies": "Filme" "movies": "Filme"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "Ausstehend", "pending": "Ausstehend",
"approved": "Genehmigt", "approved": "Genehmigt",

View file

@ -53,6 +53,11 @@
"remaining": "Remaining", "remaining": "Remaining",
"downloaded": "Downloaded" "downloaded": "Downloaded"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Active", "active": "Active",
"upload": "Upload", "upload": "Upload",
@ -68,6 +73,11 @@
"queued": "Queued", "queued": "Queued",
"movies": "Movies" "movies": "Movies"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "Pending", "pending": "Pending",
"approved": "Approved", "approved": "Approved",

View file

@ -36,6 +36,11 @@
"remaining": "Restante", "remaining": "Restante",
"downloaded": "Descargado" "downloaded": "Descargado"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Activo", "active": "Activo",
"upload": "Subir", "upload": "Subir",
@ -51,6 +56,11 @@
"queued": "Puesto en cola", "queued": "Puesto en cola",
"movies": "Películas" "movies": "Películas"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "Pendiente", "pending": "Pendiente",
"approved": "Aprobado", "approved": "Aprobado",

View file

@ -36,6 +36,11 @@
"remaining": "Restante", "remaining": "Restante",
"downloaded": "Téléchargé" "downloaded": "Téléchargé"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Active", "active": "Active",
"upload": "Téléverser", "upload": "Téléverser",
@ -51,6 +56,11 @@
"queued": "En queue", "queued": "En queue",
"movies": "Films" "movies": "Films"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "En attente", "pending": "En attente",
"approved": "Approuvée", "approved": "Approuvée",

View file

@ -51,6 +51,11 @@
"remaining": "Remaining", "remaining": "Remaining",
"downloaded": "Downloaded" "downloaded": "Downloaded"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Active", "active": "Active",
"upload": "Upload", "upload": "Upload",
@ -66,6 +71,11 @@
"queued": "Queued", "queued": "Queued",
"movies": "Movies" "movies": "Movies"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "Pending", "pending": "Pending",
"approved": "Approved", "approved": "Approved",

View file

@ -36,6 +36,11 @@
"remaining": "Gjenstående", "remaining": "Gjenstående",
"downloaded": "Nedlastet" "downloaded": "Nedlastet"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Aktiv", "active": "Aktiv",
"upload": "Opplasting", "upload": "Opplasting",
@ -51,6 +56,11 @@
"queued": "I kø", "queued": "I kø",
"movies": "Filmer" "movies": "Filmer"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "Venter", "pending": "Venter",
"approved": "Godkjent", "approved": "Godkjent",

View file

@ -21,6 +21,11 @@
"remaining": "Overgebleven", "remaining": "Overgebleven",
"downloaded": "Gedownload" "downloaded": "Gedownload"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"speedtest": { "speedtest": {
"upload": "Upload", "upload": "Upload",
"download": "Download", "download": "Download",
@ -67,6 +72,11 @@
"wanted": "Gezocht", "wanted": "Gezocht",
"queued": "In de wachtrij" "queued": "In de wachtrij"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "In afwachting", "pending": "In afwachting",
"approved": "Goedgekeurd", "approved": "Goedgekeurd",

View file

@ -36,6 +36,11 @@
"remaining": "Em falta", "remaining": "Em falta",
"downloaded": "Baixada" "downloaded": "Baixada"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Ativa", "active": "Ativa",
"upload": "Envio", "upload": "Envio",
@ -51,6 +56,11 @@
"queued": "Enfileiradas", "queued": "Enfileiradas",
"movies": "Filmes" "movies": "Filmes"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "Pendente", "pending": "Pendente",
"approved": "Aprovada", "approved": "Aprovada",

View file

@ -36,6 +36,11 @@
"remaining": "Осталось", "remaining": "Осталось",
"downloaded": "Загружено" "downloaded": "Загружено"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Активный", "active": "Активный",
"upload": "Загрузить", "upload": "Загрузить",
@ -51,6 +56,11 @@
"queued": "В очереди", "queued": "В очереди",
"movies": "Фильмы" "movies": "Фильмы"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "Ожидание", "pending": "Ожидание",
"approved": "Одобрено", "approved": "Одобрено",

View file

@ -36,6 +36,11 @@
"remaining": "Còn lại", "remaining": "Còn lại",
"downloaded": "Đã tải" "downloaded": "Đã tải"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "Hoạt động", "active": "Hoạt động",
"upload": "Tải lên", "upload": "Tải lên",
@ -51,6 +56,11 @@
"queued": "Queued", "queued": "Queued",
"movies": "Movies" "movies": "Movies"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "Pending", "pending": "Pending",
"approved": "Approved", "approved": "Approved",

View file

@ -36,6 +36,11 @@
"remaining": "其余的", "remaining": "其余的",
"downloaded": "下载" "downloaded": "下载"
}, },
"sabnzbd": {
"rate": "Rate",
"queue": "Queue",
"timeleft": "Time Left"
},
"rutorrent": { "rutorrent": {
"active": "积极的", "active": "积极的",
"upload": "上传", "upload": "上传",
@ -51,6 +56,11 @@
"queued": "排队", "queued": "排队",
"movies": "电影" "movies": "电影"
}, },
"readarr": {
"wanted": "Wanted",
"queued": "Queued",
"books": "Books"
},
"ombi": { "ombi": {
"pending": "待办的", "pending": "待办的",
"approved": "得到正式认可的", "approved": "得到正式认可的",

View file

@ -2,10 +2,12 @@ import { useTranslation } from "react-i18next";
import Sonarr from "./widgets/service/sonarr"; import Sonarr from "./widgets/service/sonarr";
import Radarr from "./widgets/service/radarr"; import Radarr from "./widgets/service/radarr";
import Readarr from "./widgets/service/readarr";
import Ombi from "./widgets/service/ombi"; import Ombi from "./widgets/service/ombi";
import Portainer from "./widgets/service/portainer"; import Portainer from "./widgets/service/portainer";
import Emby from "./widgets/service/emby"; import Emby from "./widgets/service/emby";
import Nzbget from "./widgets/service/nzbget"; import Nzbget from "./widgets/service/nzbget";
import SABnzbd from "./widgets/service/sabnzbd";
import Docker from "./widgets/service/docker"; import Docker from "./widgets/service/docker";
import Pihole from "./widgets/service/pihole"; import Pihole from "./widgets/service/pihole";
import Rutorrent from "./widgets/service/rutorrent"; import Rutorrent from "./widgets/service/rutorrent";
@ -21,11 +23,13 @@ const widgetMappings = {
docker: Docker, docker: Docker,
sonarr: Sonarr, sonarr: Sonarr,
radarr: Radarr, radarr: Radarr,
readarr: Readarr,
ombi: Ombi, ombi: Ombi,
portainer: Portainer, portainer: Portainer,
emby: Emby, emby: Emby,
jellyfin: Jellyfin, jellyfin: Jellyfin,
nzbget: Nzbget, nzbget: Nzbget,
sabnzbd: SABnzbd,
pihole: Pihole, pihole: Pihole,
rutorrent: Rutorrent, rutorrent: Rutorrent,
speedtest: Speedtest, speedtest: Speedtest,

View file

@ -0,0 +1,41 @@
import useSWR from "swr";
import { useTranslation } from "react-i18next";
import Widget from "../widget";
import Block from "../block";
import { formatApiUrl } from "utils/api-helpers";
export default function Readarr({ service }) {
const { t } = useTranslation();
const config = service.widget;
const { data: booksData, error: booksError } = useSWR(formatApiUrl(config, "book"));
const { data: wantedData, error: wantedError } = useSWR(formatApiUrl(config, "wanted/missing"));
const { data: queueData, error: queueError } = useSWR(formatApiUrl(config, "queue/status"));
if (booksError || wantedError || queueError) {
return <Widget error={t("widget.api_error")} />;
}
if (!booksData || !wantedData || !queueData) {
return (
<Widget>
<Block label={t("readarr.wanted")} />
<Block label={t("readarr.queued")} />
<Block label={t("readarr.books")} />
</Widget>
);
}
const have = booksData.filter((book) => book.statistics.bookFileCount > 0);
return (
<Widget>
<Block label={t("readarr.wanted")} value={wantedData.totalRecords} />
<Block label={t("readarr.queued")} value={queueData.totalCount} />
<Block label={t("readarr.books")} value={have.length} />
</Widget>
);
}

View file

@ -0,0 +1,37 @@
import useSWR from "swr";
import { useTranslation } from "react-i18next";
import Widget from "../widget";
import Block from "../block";
import { formatApiUrl } from "utils/api-helpers";
export default function SABnzbd({ service }) {
const { t } = useTranslation();
const config = service.widget;
const { data: queueData, error: queueError } = useSWR(formatApiUrl(config, "queue"));
if (queueError) {
return <Widget error={t("widget.api_error")} />;
}
if (!queueData) {
return (
<Widget>
<Block label={t("sabnzbd.rate")} />
<Block label={t("sabnzbd.queue")} />
<Block label={t("sabnzbd.timeleft")} />
</Widget>
);
}
return (
<Widget>
<Block label={t("sabnzbd.rate")} value={`${queueData.queue.speed}bps`} />
<Block label={t("sabnzbd.queue")} value={queueData.queue.noofslots} />
<Block label={t("sabnzbd.timeleft")} value={queueData.queue.timeleft} />
</Widget>
);
}

View file

@ -11,9 +11,11 @@ const serviceProxyHandlers = {
pihole: genericProxyHandler, pihole: genericProxyHandler,
radarr: genericProxyHandler, radarr: genericProxyHandler,
sonarr: genericProxyHandler, sonarr: genericProxyHandler,
readarr: genericProxyHandler,
speedtest: genericProxyHandler, speedtest: genericProxyHandler,
tautulli: genericProxyHandler, tautulli: genericProxyHandler,
traefik: genericProxyHandler, traefik: genericProxyHandler,
sabnzbd: genericProxyHandler,
// uses X-API-Key header auth // uses X-API-Key header auth
portainer: credentialedProxyHandler, portainer: credentialedProxyHandler,
jellyseerr: credentialedProxyHandler, jellyseerr: credentialedProxyHandler,

View file

@ -13,6 +13,8 @@ const formats = {
overseerr: `{url}/api/v1/{endpoint}`, overseerr: `{url}/api/v1/{endpoint}`,
ombi: `{url}/api/v1/{endpoint}`, ombi: `{url}/api/v1/{endpoint}`,
npm: `{url}/api/{endpoint}`, npm: `{url}/api/{endpoint}`,
readarr: `{url}/api/v1/{endpoint}?apikey={key}`,
sabnzbd: `{url}/api/?apikey={key}&output=json&mode={endpoint}`,
}; };
export function formatApiCall(api, args) { export function formatApiCall(api, args) {