mirror of
https://github.com/karl0ss/homepage.git
synced 2025-05-01 21:13:39 +01:00
Feature: Beszel service widget (#4251)
This commit is contained in:
parent
7c3dcf20ef
commit
912ae0adfc
20
docs/widgets/services/beszel.md
Normal file
20
docs/widgets/services/beszel.md
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
---
|
||||||
|
title: Beszel
|
||||||
|
description: Beszel Widget Configuration
|
||||||
|
---
|
||||||
|
|
||||||
|
Learn more about [Beszel]()
|
||||||
|
|
||||||
|
The widget has two modes, a single system with detailed info if `systemId` is provided, or an overview of all systems if `systemId` is not provided.
|
||||||
|
|
||||||
|
Allowed fields for 'overview' mode: `["systems", "up"]`
|
||||||
|
Allowed fields for a single system: `["name", "status", "updated", "cpu", "memory", "disk", "network"]`
|
||||||
|
|
||||||
|
```yaml
|
||||||
|
widget:
|
||||||
|
type: beszel
|
||||||
|
url: http://beszel.host.or.ip
|
||||||
|
username: username # email
|
||||||
|
password: password
|
||||||
|
systemId: systemId # optional
|
||||||
|
```
|
@ -14,6 +14,7 @@ You can also find a list of all available service widgets in the sidebar navigat
|
|||||||
- [Autobrr](autobrr.md)
|
- [Autobrr](autobrr.md)
|
||||||
- [Azure DevOps](azuredevops.md)
|
- [Azure DevOps](azuredevops.md)
|
||||||
- [Bazarr](bazarr.md)
|
- [Bazarr](bazarr.md)
|
||||||
|
- [Beszel](beszel.md)
|
||||||
- [Caddy](caddy.md)
|
- [Caddy](caddy.md)
|
||||||
- [Calendar](calendar.md)
|
- [Calendar](calendar.md)
|
||||||
- [Calibre-Web](calibre-web.md)
|
- [Calibre-Web](calibre-web.md)
|
||||||
|
@ -37,6 +37,7 @@ nav:
|
|||||||
- widgets/services/autobrr.md
|
- widgets/services/autobrr.md
|
||||||
- widgets/services/azuredevops.md
|
- widgets/services/azuredevops.md
|
||||||
- widgets/services/bazarr.md
|
- widgets/services/bazarr.md
|
||||||
|
- widgets/services/beszel.md
|
||||||
- widgets/services/caddy.md
|
- widgets/services/caddy.md
|
||||||
- widgets/services/calendar.md
|
- widgets/services/calendar.md
|
||||||
- widgets/services/calibre-web.md
|
- widgets/services/calibre-web.md
|
||||||
|
@ -967,5 +967,16 @@
|
|||||||
"status": "Status",
|
"status": "Status",
|
||||||
"online": "Online",
|
"online": "Online",
|
||||||
"offline": "Offline"
|
"offline": "Offline"
|
||||||
|
},
|
||||||
|
"beszel": {
|
||||||
|
"name": "Name",
|
||||||
|
"systems": "Systems",
|
||||||
|
"up": "Up",
|
||||||
|
"status": "Status",
|
||||||
|
"updated": "Updated",
|
||||||
|
"cpu": "CPU",
|
||||||
|
"memory": "MEM",
|
||||||
|
"disk": "Disk",
|
||||||
|
"network": "NET"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -368,6 +368,9 @@ export function cleanServiceGroups(groups) {
|
|||||||
repositoryId,
|
repositoryId,
|
||||||
userEmail,
|
userEmail,
|
||||||
|
|
||||||
|
// beszel
|
||||||
|
systemId,
|
||||||
|
|
||||||
// calendar
|
// calendar
|
||||||
firstDayInWeek,
|
firstDayInWeek,
|
||||||
integrations,
|
integrations,
|
||||||
@ -511,6 +514,10 @@ export function cleanServiceGroups(groups) {
|
|||||||
if (repositoryId) cleanedService.widget.repositoryId = repositoryId;
|
if (repositoryId) cleanedService.widget.repositoryId = repositoryId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (type === "beszel") {
|
||||||
|
if (systemId) cleanedService.widget.systemId = systemId;
|
||||||
|
}
|
||||||
|
|
||||||
if (type === "coinmarketcap") {
|
if (type === "coinmarketcap") {
|
||||||
if (currency) cleanedService.widget.currency = currency;
|
if (currency) cleanedService.widget.currency = currency;
|
||||||
if (symbols) cleanedService.widget.symbols = symbols;
|
if (symbols) cleanedService.widget.symbols = symbols;
|
||||||
|
60
src/widgets/beszel/component.jsx
Normal file
60
src/widgets/beszel/component.jsx
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
import { useTranslation } from "next-i18next";
|
||||||
|
|
||||||
|
import Container from "components/services/widget/container";
|
||||||
|
import Block from "components/services/widget/block";
|
||||||
|
import useWidgetAPI from "utils/proxy/use-widget-api";
|
||||||
|
|
||||||
|
export default function Component({ service }) {
|
||||||
|
const { t } = useTranslation();
|
||||||
|
|
||||||
|
const { widget } = service;
|
||||||
|
const { systemId } = widget;
|
||||||
|
|
||||||
|
const { data: systems, error: systemsError } = useWidgetAPI(widget, "systems");
|
||||||
|
|
||||||
|
const MAX_ALLOWED_FIELDS = 4;
|
||||||
|
if (!widget.fields?.length > 0) {
|
||||||
|
widget.fields = systemId ? ["name", "status", "cpu", "memory"] : ["systems", "up"];
|
||||||
|
}
|
||||||
|
if (widget.fields?.length > MAX_ALLOWED_FIELDS) {
|
||||||
|
widget.fields = widget.fields.slice(0, MAX_ALLOWED_FIELDS);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (systemsError) {
|
||||||
|
return <Container service={service} error={systemsError} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!systems) {
|
||||||
|
return (
|
||||||
|
<Container service={service}>
|
||||||
|
<Block label="beszel.systems" />
|
||||||
|
<Block label="beszel.up" />
|
||||||
|
</Container>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (systemId) {
|
||||||
|
const system = systems.items.find((item) => item.id === systemId);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Container service={service}>
|
||||||
|
<Block label="beszel.name" value={system.name} />
|
||||||
|
<Block label="beszel.status" value={t(`beszel.${system.status}`)} />
|
||||||
|
<Block label="beszel.updated" value={t("common.relativeDate", { value: system.updated })} />
|
||||||
|
<Block label="beszel.cpu" value={t("common.percent", { value: system.info.cpu, maximumFractionDigits: 2 })} />
|
||||||
|
<Block label="beszel.memory" value={t("common.percent", { value: system.info.mp, maximumFractionDigits: 2 })} />
|
||||||
|
<Block label="beszel.disk" value={t("common.percent", { value: system.info.dp, maximumFractionDigits: 2 })} />
|
||||||
|
<Block label="beszel.network" value={t("common.percent", { value: system.info.b, maximumFractionDigits: 2 })} />
|
||||||
|
</Container>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const upTotal = systems.items.filter((item) => item.status === "up").length;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Container service={service}>
|
||||||
|
<Block label="beszel.systems" value={systems.totalItems} />
|
||||||
|
<Block label="beszel.up" value={`${upTotal} / ${systems.totalItems}`} />
|
||||||
|
</Container>
|
||||||
|
);
|
||||||
|
}
|
99
src/widgets/beszel/proxy.js
Normal file
99
src/widgets/beszel/proxy.js
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
import cache from "memory-cache";
|
||||||
|
|
||||||
|
import getServiceWidget from "utils/config/service-helpers";
|
||||||
|
import { formatApiCall } from "utils/proxy/api-helpers";
|
||||||
|
import { httpProxy } from "utils/proxy/http";
|
||||||
|
import widgets from "widgets/widgets";
|
||||||
|
import createLogger from "utils/logger";
|
||||||
|
|
||||||
|
const proxyName = "beszelProxyHandler";
|
||||||
|
const tokenCacheKey = `${proxyName}__token`;
|
||||||
|
const logger = createLogger(proxyName);
|
||||||
|
|
||||||
|
async function login(loginUrl, username, password, service) {
|
||||||
|
const authResponse = await httpProxy(loginUrl, {
|
||||||
|
method: "POST",
|
||||||
|
body: JSON.stringify({ identity: username, password }),
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const status = authResponse[0];
|
||||||
|
let data = authResponse[2];
|
||||||
|
try {
|
||||||
|
data = JSON.parse(Buffer.from(authResponse[2]).toString());
|
||||||
|
|
||||||
|
if (status === 200) {
|
||||||
|
cache.put(`${tokenCacheKey}.${service}`, data.token);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
logger.error(`Error ${status} logging into beszel`, JSON.stringify(authResponse[2]));
|
||||||
|
}
|
||||||
|
return [status, data.token ?? data];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async function beszelProxyHandler(req, res) {
|
||||||
|
const { group, service, endpoint } = req.query;
|
||||||
|
|
||||||
|
if (group && service) {
|
||||||
|
const widget = await getServiceWidget(group, service);
|
||||||
|
|
||||||
|
if (!widgets?.[widget.type]?.api) {
|
||||||
|
return res.status(403).json({ error: "Service does not support API calls" });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (widget) {
|
||||||
|
const url = new URL(formatApiCall(widgets[widget.type].api, { endpoint, ...widget }));
|
||||||
|
const loginUrl = formatApiCall(widgets[widget.type].api, { endpoint: "admins/auth-with-password", ...widget });
|
||||||
|
|
||||||
|
let status;
|
||||||
|
let data;
|
||||||
|
|
||||||
|
let token = cache.get(`${tokenCacheKey}.${service}`);
|
||||||
|
if (!token) {
|
||||||
|
[status, token] = await login(loginUrl, widget.username, widget.password, service);
|
||||||
|
if (status !== 200) {
|
||||||
|
logger.debug(`HTTTP ${status} logging into npm api: ${token}`);
|
||||||
|
return res.status(status).send(token);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
[status, , data] = await httpProxy(url, {
|
||||||
|
method: "GET",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
Authorization: `Bearer ${token}`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (status === 403) {
|
||||||
|
logger.debug(`HTTTP ${status} retrieving data from npm api, logging in and trying again.`);
|
||||||
|
cache.del(`${tokenCacheKey}.${service}`);
|
||||||
|
[status, token] = await login(loginUrl, widget.username, widget.password, service);
|
||||||
|
|
||||||
|
if (status !== 200) {
|
||||||
|
logger.debug(`HTTTP ${status} logging into npm api: ${data}`);
|
||||||
|
return res.status(status).send(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
[status, , data] = await httpProxy(url, {
|
||||||
|
method: "GET",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
Authorization: `Bearer ${token}`,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (status !== 200) {
|
||||||
|
return res.status(status).send(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.send(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.status(400).json({ error: "Invalid proxy service type" });
|
||||||
|
}
|
14
src/widgets/beszel/widget.js
Normal file
14
src/widgets/beszel/widget.js
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import beszelProxyHandler from "./proxy";
|
||||||
|
|
||||||
|
const widget = {
|
||||||
|
api: "{url}/api/{endpoint}",
|
||||||
|
proxyHandler: beszelProxyHandler,
|
||||||
|
|
||||||
|
mappings: {
|
||||||
|
systems: {
|
||||||
|
endpoint: "collections/systems/records?page=1&perPage=500&sort=%2Bcreated",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default widget;
|
@ -8,6 +8,7 @@ const components = {
|
|||||||
autobrr: dynamic(() => import("./autobrr/component")),
|
autobrr: dynamic(() => import("./autobrr/component")),
|
||||||
azuredevops: dynamic(() => import("./azuredevops/component")),
|
azuredevops: dynamic(() => import("./azuredevops/component")),
|
||||||
bazarr: dynamic(() => import("./bazarr/component")),
|
bazarr: dynamic(() => import("./bazarr/component")),
|
||||||
|
beszel: dynamic(() => import("./beszel/component")),
|
||||||
caddy: dynamic(() => import("./caddy/component")),
|
caddy: dynamic(() => import("./caddy/component")),
|
||||||
calendar: dynamic(() => import("./calendar/component")),
|
calendar: dynamic(() => import("./calendar/component")),
|
||||||
calibreweb: dynamic(() => import("./calibreweb/component")),
|
calibreweb: dynamic(() => import("./calibreweb/component")),
|
||||||
|
@ -5,6 +5,7 @@ import authentik from "./authentik/widget";
|
|||||||
import autobrr from "./autobrr/widget";
|
import autobrr from "./autobrr/widget";
|
||||||
import azuredevops from "./azuredevops/widget";
|
import azuredevops from "./azuredevops/widget";
|
||||||
import bazarr from "./bazarr/widget";
|
import bazarr from "./bazarr/widget";
|
||||||
|
import beszel from "./beszel/widget";
|
||||||
import caddy from "./caddy/widget";
|
import caddy from "./caddy/widget";
|
||||||
import calendar from "./calendar/widget";
|
import calendar from "./calendar/widget";
|
||||||
import calibreweb from "./calibreweb/widget";
|
import calibreweb from "./calibreweb/widget";
|
||||||
@ -133,6 +134,7 @@ const widgets = {
|
|||||||
autobrr,
|
autobrr,
|
||||||
azuredevops,
|
azuredevops,
|
||||||
bazarr,
|
bazarr,
|
||||||
|
beszel,
|
||||||
caddy,
|
caddy,
|
||||||
calibreweb,
|
calibreweb,
|
||||||
changedetectionio,
|
changedetectionio,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user