Merge pull request #116 from JazzFisch/additional-widgets
Add Readarr and SABnzbd widgets
This commit is contained in:
commit
ee79335eff
17 changed files with 197 additions and 0 deletions
|
@ -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))
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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": "Одобрено",
|
||||||
|
|
|
@ -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",
|
||||||
|
|
|
@ -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": "得到正式认可的",
|
||||||
|
|
|
@ -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,
|
||||||
|
|
41
src/components/services/widgets/service/readarr.jsx
Normal file
41
src/components/services/widgets/service/readarr.jsx
Normal 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>
|
||||||
|
);
|
||||||
|
}
|
37
src/components/services/widgets/service/sabnzbd.jsx
Normal file
37
src/components/services/widgets/service/sabnzbd.jsx
Normal 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>
|
||||||
|
);
|
||||||
|
}
|
|
@ -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,
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
Loading…
Add table
Reference in a new issue