Add Seafile widget
This commit is contained in:
parent
1d3aba973e
commit
fee57a59e9
@ -625,5 +625,11 @@
|
|||||||
"repos": "Repositories",
|
"repos": "Repositories",
|
||||||
"users": "Users",
|
"users": "Users",
|
||||||
"orgs": "Organizations"
|
"orgs": "Organizations"
|
||||||
|
},
|
||||||
|
"seafile": {
|
||||||
|
"users": "Users",
|
||||||
|
"groups": "Groups",
|
||||||
|
"libraries": "Libraries",
|
||||||
|
"storage": "Storage"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -70,6 +70,7 @@ const components = {
|
|||||||
rutorrent: dynamic(() => import("./rutorrent/component")),
|
rutorrent: dynamic(() => import("./rutorrent/component")),
|
||||||
sabnzbd: dynamic(() => import("./sabnzbd/component")),
|
sabnzbd: dynamic(() => import("./sabnzbd/component")),
|
||||||
scrutiny: dynamic(() => import("./scrutiny/component")),
|
scrutiny: dynamic(() => import("./scrutiny/component")),
|
||||||
|
seafile: dynamic(() => import("./seafile/component")),
|
||||||
shoko: dynamic(() => import("./shoko/component")),
|
shoko: dynamic(() => import("./shoko/component")),
|
||||||
sonarr: dynamic(() => import("./sonarr/component")),
|
sonarr: dynamic(() => import("./sonarr/component")),
|
||||||
speedtest: dynamic(() => import("./speedtest/component")),
|
speedtest: dynamic(() => import("./speedtest/component")),
|
||||||
|
|||||||
40
src/widgets/seafile/component.jsx
Normal file
40
src/widgets/seafile/component.jsx
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
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 { data: seafileData, error: seafileError } = useWidgetAPI(widget);
|
||||||
|
|
||||||
|
if (seafileError) {
|
||||||
|
return <Container service={service} error={seafileError} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!seafileData) {
|
||||||
|
return (
|
||||||
|
<Container service={service}>
|
||||||
|
<Block label="seafile.users" />
|
||||||
|
<Block label="seafile.groups" />
|
||||||
|
<Block label="seafile.libraries" />
|
||||||
|
<Block label="seafile.storage" />
|
||||||
|
</Container>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Container service={service}>
|
||||||
|
<Block label="seafile.users" value={t("common.number", { value: seafileData.users_count })} />
|
||||||
|
<Block label="seafile.groups" value={t("common.number", { value: seafileData.groups_count })} />
|
||||||
|
<Block label="seafile.libraries" value={t("common.number", { value: seafileData.repos_count })} />
|
||||||
|
<Block
|
||||||
|
label="seafile.storage"
|
||||||
|
value={t("common.bytes", { value: seafileData.total_storage, maximumFractionDigits: 1 })}
|
||||||
|
/>
|
||||||
|
</Container>
|
||||||
|
);
|
||||||
|
}
|
||||||
82
src/widgets/seafile/proxy.js
Normal file
82
src/widgets/seafile/proxy.js
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
import cache from "memory-cache";
|
||||||
|
|
||||||
|
import { formatApiCall } from "utils/proxy/api-helpers";
|
||||||
|
import { httpProxy } from "utils/proxy/http";
|
||||||
|
import getServiceWidget from "utils/config/service-helpers";
|
||||||
|
import createLogger from "utils/logger";
|
||||||
|
import widgets from "widgets/widgets";
|
||||||
|
|
||||||
|
const proxyName = "seafileProxyHandler";
|
||||||
|
const sessionTokenCacheKey = `${proxyName}__sessionToken`;
|
||||||
|
const logger = createLogger(proxyName);
|
||||||
|
|
||||||
|
async function login(widget, service) {
|
||||||
|
const endpoint = "api2/auth-token/";
|
||||||
|
const api = widgets?.[widget.type]?.api;
|
||||||
|
const loginUrl = new URL(formatApiCall(api, { endpoint, ...widget }));
|
||||||
|
const loginParams = {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/x-www-form-urlencoded",
|
||||||
|
},
|
||||||
|
body: new URLSearchParams({
|
||||||
|
username: widget.username,
|
||||||
|
password: widget.password,
|
||||||
|
}).toString(),
|
||||||
|
};
|
||||||
|
|
||||||
|
// eslint-disable-next-line no-unused-vars
|
||||||
|
const [status, contentType, data] = await httpProxy(loginUrl, loginParams);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const { token } = JSON.parse(data.toString());
|
||||||
|
cache.put(`${sessionTokenCacheKey}.${service}`, token);
|
||||||
|
return { token };
|
||||||
|
} catch (e) {
|
||||||
|
logger.error("Unable to login to Seafile API: %s", e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return { token: false };
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async function seafileProxyHandler(req, res) {
|
||||||
|
const { group, service } = req.query;
|
||||||
|
|
||||||
|
if (!group || !service) {
|
||||||
|
logger.debug("Invalid or missing service '%s' or group '%s'", service, group);
|
||||||
|
return res.status(400).json({ error: "Invalid proxy service type" });
|
||||||
|
}
|
||||||
|
|
||||||
|
const widget = await getServiceWidget(group, service);
|
||||||
|
|
||||||
|
if (!widget) {
|
||||||
|
logger.debug("Invalid or missing widget for service '%s' in group '%s'", service, group);
|
||||||
|
return res.status(400).json({ error: "Invalid proxy service type" });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!cache.get(`${sessionTokenCacheKey}.${service}`)) {
|
||||||
|
await login(widget, service);
|
||||||
|
}
|
||||||
|
|
||||||
|
const endpoint = "api/v2.1/admin/sysinfo/";
|
||||||
|
const url = new URL(formatApiCall(widgets[widget.type].api, { endpoint, ...widget }));
|
||||||
|
const params = {
|
||||||
|
method: "GET",
|
||||||
|
headers: {
|
||||||
|
Authorization: `Token ${cache.get(`${sessionTokenCacheKey}.${service}`)}`,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
let [status, contentType, data] = await httpProxy(url, params);
|
||||||
|
|
||||||
|
if (status === 401) {
|
||||||
|
logger.debug("Seafile API rejected the request, attempting to obtain new session token");
|
||||||
|
const { token } = await login(widget, service);
|
||||||
|
params.headers.Authorization = `Token ${token}`;
|
||||||
|
|
||||||
|
[status, contentType, data] = await httpProxy(url, params);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (contentType) res.setHeader("Content-Type", contentType);
|
||||||
|
return res.status(status).send(data);
|
||||||
|
}
|
||||||
8
src/widgets/seafile/widget.js
Normal file
8
src/widgets/seafile/widget.js
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
import seafileProxyHandler from "./proxy";
|
||||||
|
|
||||||
|
const widget = {
|
||||||
|
api: "{url}/{endpoint}",
|
||||||
|
proxyHandler: seafileProxyHandler,
|
||||||
|
};
|
||||||
|
|
||||||
|
export default widget;
|
||||||
@ -64,6 +64,7 @@ import readarr from "./readarr/widget";
|
|||||||
import rutorrent from "./rutorrent/widget";
|
import rutorrent from "./rutorrent/widget";
|
||||||
import sabnzbd from "./sabnzbd/widget";
|
import sabnzbd from "./sabnzbd/widget";
|
||||||
import scrutiny from "./scrutiny/widget";
|
import scrutiny from "./scrutiny/widget";
|
||||||
|
import seafile from "./seafile/widget";
|
||||||
import shoko from "./shoko/widget";
|
import shoko from "./shoko/widget";
|
||||||
import sonarr from "./sonarr/widget";
|
import sonarr from "./sonarr/widget";
|
||||||
import speedtest from "./speedtest/widget";
|
import speedtest from "./speedtest/widget";
|
||||||
@ -150,6 +151,7 @@ const widgets = {
|
|||||||
rutorrent,
|
rutorrent,
|
||||||
sabnzbd,
|
sabnzbd,
|
||||||
scrutiny,
|
scrutiny,
|
||||||
|
seafile,
|
||||||
shoko,
|
shoko,
|
||||||
sonarr,
|
sonarr,
|
||||||
speedtest,
|
speedtest,
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user