forked from gethomepage/homepage
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Feature: Wg-Easy Widget (gethomepage#3476)
--------- Co-authored-by: ConnerWithAnE <[email protected]> Co-authored-by: shamoon <[email protected]>
- Loading branch information
1 parent
1144f4d
commit 6ab6d6f
Showing
8 changed files
with
158 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
--- | ||
title: Wg-Easy | ||
description: Wg-Easy Widget Configuration | ||
--- | ||
|
||
Learn more about [Wg-Easy](https://github.com/wg-easy/wg-easy). | ||
|
||
Allowed fields: `["connected", "enabled", "disabled", "total"]`. | ||
|
||
Note: by default `["connected", "enabled", "total"]` are displayed. | ||
|
||
To detect if a device is connected the time since the last handshake is queried. `threshold` is the time to wait in minutes since the last handshake to consider a device connected. Default is 2 minutes. | ||
|
||
```yaml | ||
widget: | ||
type: wgeasy | ||
url: http://wg.easy.or.ip | ||
password: yourwgeasypassword | ||
threshold: 2 # optional | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,45 @@ | ||
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 { widget } = service; | ||
|
||
const { data: infoData, error: infoError } = useWidgetAPI(widget); | ||
|
||
if (!widget.fields) { | ||
widget.fields = ["connected", "enabled", "total"]; | ||
} | ||
|
||
if (infoError) { | ||
return <Container service={service} error={infoError} />; | ||
} | ||
|
||
if (!infoData) { | ||
return ( | ||
<Container service={service}> | ||
<Block label="wgeasy.connected" /> | ||
<Block label="wgeasy.enabled" /> | ||
<Block label="wgeasy.disabled" /> | ||
<Block label="wgeasy.total" /> | ||
</Container> | ||
); | ||
} | ||
|
||
const enabled = infoData.filter((item) => item.enabled).length; | ||
const disabled = infoData.length - enabled; | ||
const connectionThreshold = widget.threshold ?? 2 * 60 * 1000; | ||
const currentTime = new Date(); | ||
const connected = infoData.filter( | ||
(item) => currentTime - new Date(item.latestHandshakeAt) < connectionThreshold, | ||
).length; | ||
|
||
return ( | ||
<Container service={service}> | ||
<Block label="wgeasy.connected" value={connected} /> | ||
<Block label="wgeasy.enabled" value={enabled} /> | ||
<Block label="wgeasy.diabled" value={disabled} /> | ||
<Block label="wgeasy.total" value={infoData.length} /> | ||
</Container> | ||
); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,70 @@ | ||
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 = "wgeasyProxyHandler"; | ||
const logger = createLogger(proxyName); | ||
const sessionSIDCacheKey = `${proxyName}__sessionSID`; | ||
|
||
async function login(widget, service) { | ||
const url = formatApiCall(widgets[widget.type].api, { ...widget, endpoint: "session" }); | ||
const [, , , responseHeaders] = await httpProxy(url, { | ||
method: "POST", | ||
body: JSON.stringify({ password: widget.password }), | ||
headers: { | ||
"Content-Type": "application/json", | ||
}, | ||
}); | ||
|
||
try { | ||
const connectSidCookie = responseHeaders["set-cookie"] | ||
.find((cookie) => cookie.startsWith("connect.sid=")) | ||
.split(";")[0] | ||
.replace("connect.sid=", ""); | ||
cache.put(`${sessionSIDCacheKey}.${service}`, connectSidCookie); | ||
return connectSidCookie; | ||
} catch (e) { | ||
logger.error(`Error logging into wg-easy`); | ||
cache.del(`${sessionSIDCacheKey}.${service}`); | ||
return null; | ||
} | ||
} | ||
|
||
export default async function wgeasyProxyHandler(req, res) { | ||
const { group, service } = 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) { | ||
let sid = cache.get(`${sessionSIDCacheKey}.${service}`); | ||
if (!sid) { | ||
sid = await login(widget, service); | ||
if (!sid) { | ||
return res.status(500).json({ error: "Failed to authenticate with Wg-Easy" }); | ||
} | ||
} | ||
const [, , data] = await httpProxy( | ||
formatApiCall(widgets[widget.type].api, { ...widget, endpoint: "wireguard/client" }), | ||
{ | ||
headers: { | ||
"Content-Type": "application/json", | ||
Cookie: `connect.sid=${sid}`, | ||
}, | ||
}, | ||
); | ||
|
||
return res.json(JSON.parse(data)); | ||
} | ||
} | ||
|
||
return res.status(400).json({ error: "Invalid proxy service type" }); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
import wgeasyProxyHandler from "./proxy"; | ||
|
||
const widget = { | ||
api: "{url}/api/{endpoint}", | ||
proxyHandler: wgeasyProxyHandler, | ||
}; | ||
|
||
export default widget; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters