Add pools option to TrueNAS service widget
This commit is contained in:
parent
d20cdbb9ab
commit
a7582c4d92
@ -9,6 +9,8 @@ Allowed fields: `["load", "uptime", "alerts"]`.
|
|||||||
|
|
||||||
To create an API Key, follow [the official TrueNAS documentation](https://www.truenas.com/docs/scale/scaletutorials/toptoolbar/managingapikeys/).
|
To create an API Key, follow [the official TrueNAS documentation](https://www.truenas.com/docs/scale/scaletutorials/toptoolbar/managingapikeys/).
|
||||||
|
|
||||||
|
A detailed pool listing is disabled by default, but can be enabled with the `enablePools` option.
|
||||||
|
|
||||||
```yaml
|
```yaml
|
||||||
widget:
|
widget:
|
||||||
type: truenas
|
type: truenas
|
||||||
@ -16,4 +18,5 @@ widget:
|
|||||||
username: user # not required if using api key
|
username: user # not required if using api key
|
||||||
password: pass # not required if using api key
|
password: pass # not required if using api key
|
||||||
key: yourtruenasapikey # not required if using username / password
|
key: yourtruenasapikey # not required if using username / password
|
||||||
|
enablePools: true # optional, defaults to false
|
||||||
```
|
```
|
||||||
|
|||||||
31
src/components/widgets/truenas/pool.jsx
Normal file
31
src/components/widgets/truenas/pool.jsx
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
import classNames from "classnames";
|
||||||
|
import prettyBytes from "pretty-bytes";
|
||||||
|
|
||||||
|
export default function Pool({ name, free, used, warning }) {
|
||||||
|
const total = free + used;
|
||||||
|
const usedPercent = Math.round((used / total) * 100);
|
||||||
|
const statusColor = warning ? "bg-yellow-500" : "bg-green-500";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex flex-row text-theme-700 dark:text-theme-200 items-center text-xs relative h-5 w-full rounded-md bg-theme-200/50 dark:bg-theme-900/20 mt-1">
|
||||||
|
<div
|
||||||
|
className="absolute h-5 rounded-md bg-theme-200 dark:bg-theme-900/40 z-0"
|
||||||
|
style={{
|
||||||
|
width: `${usedPercent}%`,
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<span className="ml-2 h-2 w-2">
|
||||||
|
<span className={classNames("block w-2 h-2 rounded", statusColor)} />
|
||||||
|
</span>
|
||||||
|
<div className="text-xs z-10 self-center ml-2 relative h-4 grow mr-2">
|
||||||
|
<div className="absolute w-full whitespace-nowrap text-ellipsis overflow-hidden text-left">{name}</div>
|
||||||
|
</div>
|
||||||
|
<div className="self-center text-xs flex justify-end mr-1.5 pl-1 z-10 text-ellipsis overflow-hidden whitespace-nowrap">
|
||||||
|
<span>
|
||||||
|
{prettyBytes(used)} / {prettyBytes(total)}
|
||||||
|
</span>
|
||||||
|
<span className="pl-2 w-12 text-center">({usedPercent}%)</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@ -438,6 +438,9 @@ export function cleanServiceGroups(groups) {
|
|||||||
// sonarr, radarr
|
// sonarr, radarr
|
||||||
enableQueue,
|
enableQueue,
|
||||||
|
|
||||||
|
// truenas
|
||||||
|
enablePools,
|
||||||
|
|
||||||
// unifi
|
// unifi
|
||||||
site,
|
site,
|
||||||
} = cleanedService.widget;
|
} = cleanedService.widget;
|
||||||
@ -507,6 +510,9 @@ export function cleanServiceGroups(groups) {
|
|||||||
if (["sonarr", "radarr"].includes(type)) {
|
if (["sonarr", "radarr"].includes(type)) {
|
||||||
if (enableQueue !== undefined) cleanedService.widget.enableQueue = JSON.parse(enableQueue);
|
if (enableQueue !== undefined) cleanedService.widget.enableQueue = JSON.parse(enableQueue);
|
||||||
}
|
}
|
||||||
|
if (type === "truenas") {
|
||||||
|
if (enablePools !== undefined) cleanedService.widget.enablePools = JSON.parse(enablePools);
|
||||||
|
}
|
||||||
if (["diskstation", "qnap"].includes(type)) {
|
if (["diskstation", "qnap"].includes(type)) {
|
||||||
if (volume) cleanedService.widget.volume = volume;
|
if (volume) cleanedService.widget.volume = volume;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -70,6 +70,7 @@ export default async function credentialedProxyHandler(req, res, map) {
|
|||||||
withCredentials: true,
|
withCredentials: true,
|
||||||
credentials: "include",
|
credentials: "include",
|
||||||
headers,
|
headers,
|
||||||
|
body: req.body,
|
||||||
});
|
});
|
||||||
|
|
||||||
let resultData = data;
|
let resultData = data;
|
||||||
|
|||||||
@ -3,6 +3,7 @@ import { useTranslation } from "next-i18next";
|
|||||||
import Container from "components/services/widget/container";
|
import Container from "components/services/widget/container";
|
||||||
import Block from "components/services/widget/block";
|
import Block from "components/services/widget/block";
|
||||||
import useWidgetAPI from "utils/proxy/use-widget-api";
|
import useWidgetAPI from "utils/proxy/use-widget-api";
|
||||||
|
import Pool from "components/widgets/truenas/pool";
|
||||||
|
|
||||||
export default function Component({ service }) {
|
export default function Component({ service }) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@ -11,9 +12,10 @@ export default function Component({ service }) {
|
|||||||
|
|
||||||
const { data: alertData, error: alertError } = useWidgetAPI(widget, "alerts");
|
const { data: alertData, error: alertError } = useWidgetAPI(widget, "alerts");
|
||||||
const { data: statusData, error: statusError } = useWidgetAPI(widget, "status");
|
const { data: statusData, error: statusError } = useWidgetAPI(widget, "status");
|
||||||
|
const { data: poolsData, error: poolsError } = useWidgetAPI(widget, "pools");
|
||||||
|
|
||||||
if (alertError || statusError) {
|
if (alertError || statusError || poolsError) {
|
||||||
const finalError = alertError ?? statusError;
|
const finalError = alertError ?? statusError ?? poolsError;
|
||||||
return <Container service={service} error={finalError} />;
|
return <Container service={service} error={finalError} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -27,11 +29,19 @@ export default function Component({ service }) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const enablePools = widget?.enablePools && Array.isArray(poolsData) && poolsData.length > 0;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
<>
|
||||||
<Container service={service}>
|
<Container service={service}>
|
||||||
<Block label="truenas.load" value={t("common.number", { value: statusData.loadavg[0] })} />
|
<Block label="truenas.load" value={t("common.number", { value: statusData.loadavg[0] })} />
|
||||||
<Block label="truenas.uptime" value={t("common.uptime", { value: statusData.uptime_seconds })} />
|
<Block label="truenas.uptime" value={t("common.uptime", { value: statusData.uptime_seconds })} />
|
||||||
<Block label="truenas.alerts" value={t("common.number", { value: alertData.pending })} />
|
<Block label="truenas.alerts" value={t("common.number", { value: alertData.pending })} />
|
||||||
</Container>
|
</Container>
|
||||||
|
{enablePools &&
|
||||||
|
poolsData.map((pool) => (
|
||||||
|
<Pool key={pool.id} name={pool.name} free={pool.free} used={pool.used} warning={pool.warning} />
|
||||||
|
))}
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
29
src/widgets/truenas/proxy.js
Normal file
29
src/widgets/truenas/proxy.js
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
import getServiceWidget from "utils/config/service-helpers";
|
||||||
|
import createLogger from "utils/logger";
|
||||||
|
import genericProxyHandler from "utils/proxy/handlers/generic";
|
||||||
|
import credentialedProxyHandler from "utils/proxy/handlers/credentialed";
|
||||||
|
|
||||||
|
const logger = createLogger("truenasProxyHandler");
|
||||||
|
|
||||||
|
export default async function truenasProxyHandler(req, res, map) {
|
||||||
|
const { group, service } = req.query;
|
||||||
|
|
||||||
|
if (group && service) {
|
||||||
|
const widgetOpts = await getServiceWidget(group, service);
|
||||||
|
let handler;
|
||||||
|
if (widgetOpts.username && widgetOpts.password) {
|
||||||
|
handler = genericProxyHandler;
|
||||||
|
} else if (widgetOpts.key) {
|
||||||
|
handler = credentialedProxyHandler;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (handler) {
|
||||||
|
return handler(req, res, map);
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.error("Error getting data from Truenas: Username / password or API key required");
|
||||||
|
return res.status(500).json({ error: "Username / password or API key required" });
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.status(500).json({ error: "Error parsing widget request" });
|
||||||
|
}
|
||||||
@ -1,32 +1,10 @@
|
|||||||
import { jsonArrayFilter } from "utils/proxy/api-helpers";
|
import truenasProxyHandler from "./proxy";
|
||||||
import credentialedProxyHandler from "utils/proxy/handlers/credentialed";
|
|
||||||
import genericProxyHandler from "utils/proxy/handlers/generic";
|
import { asJson, jsonArrayFilter } from "utils/proxy/api-helpers";
|
||||||
import getServiceWidget from "utils/config/service-helpers";
|
|
||||||
|
|
||||||
const widget = {
|
const widget = {
|
||||||
api: "{url}/api/v2.0/{endpoint}",
|
api: "{url}/api/v2.0/{endpoint}",
|
||||||
proxyHandler: async (req, res, map) => {
|
proxyHandler: truenasProxyHandler,
|
||||||
// choose proxy handler based on widget settings
|
|
||||||
const { group, service } = req.query;
|
|
||||||
|
|
||||||
if (group && service) {
|
|
||||||
const widgetOpts = await getServiceWidget(group, service);
|
|
||||||
let handler;
|
|
||||||
if (widgetOpts.username && widgetOpts.password) {
|
|
||||||
handler = genericProxyHandler;
|
|
||||||
} else if (widgetOpts.key) {
|
|
||||||
handler = credentialedProxyHandler;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (handler) {
|
|
||||||
return handler(req, res, map);
|
|
||||||
}
|
|
||||||
|
|
||||||
return res.status(500).json({ error: "Username / password or API key required" });
|
|
||||||
}
|
|
||||||
|
|
||||||
return res.status(500).json({ error: "Error parsing widget request" });
|
|
||||||
},
|
|
||||||
|
|
||||||
mappings: {
|
mappings: {
|
||||||
alerts: {
|
alerts: {
|
||||||
@ -39,6 +17,19 @@ const widget = {
|
|||||||
endpoint: "system/info",
|
endpoint: "system/info",
|
||||||
validate: ["loadavg", "uptime_seconds"],
|
validate: ["loadavg", "uptime_seconds"],
|
||||||
},
|
},
|
||||||
|
pools: {
|
||||||
|
endpoint: "pool",
|
||||||
|
map: (data) =>
|
||||||
|
asJson(data).map((entry) => ({
|
||||||
|
id: entry.name,
|
||||||
|
name: entry.name,
|
||||||
|
status: entry.status,
|
||||||
|
healthy: entry.healthy,
|
||||||
|
used: entry.allocated,
|
||||||
|
size: entry.size,
|
||||||
|
free: entry.free,
|
||||||
|
})),
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user