mirror of
https://github.com/Koenkk/zigbee2mqtt.git
synced 2026-06-23 05:31:47 +00:00
239 lines
9.8 KiB
TypeScript
239 lines
9.8 KiB
TypeScript
import assert from "node:assert";
|
|
import {existsSync, readFileSync} from "node:fs";
|
|
import type {IncomingMessage, Server, ServerResponse} from "node:http";
|
|
import {createServer} from "node:http";
|
|
import {createServer as createSecureServer} from "node:https";
|
|
import type {Socket} from "node:net";
|
|
import {posix} from "node:path";
|
|
import bind from "bind-decorator";
|
|
import expressStaticGzip from "express-static-gzip";
|
|
import finalhandler from "finalhandler";
|
|
import stringify from "json-stable-stringify-without-jsonify";
|
|
import WebSocket from "ws";
|
|
|
|
import data from "../util/data";
|
|
import logger from "../util/logger";
|
|
import * as settings from "../util/settings";
|
|
import utils from "../util/utils";
|
|
import Extension from "./extension";
|
|
|
|
/**
|
|
* This extension servers the frontend
|
|
*/
|
|
export class Frontend extends Extension {
|
|
private mqttBaseTopic: string;
|
|
private server: Server | undefined;
|
|
private wss!: WebSocket.Server;
|
|
private baseUrl: string;
|
|
|
|
constructor(
|
|
zigbee: Zigbee,
|
|
mqtt: Mqtt,
|
|
state: State,
|
|
publishEntityState: PublishEntityState,
|
|
eventBus: EventBus,
|
|
enableDisableExtension: (enable: boolean, name: string) => Promise<void>,
|
|
restartCallback: () => Promise<void>,
|
|
addExtension: (extension: Extension) => Promise<void>,
|
|
) {
|
|
super(zigbee, mqtt, state, publishEntityState, eventBus, enableDisableExtension, restartCallback, addExtension);
|
|
|
|
const frontendSettings = settings.get().frontend;
|
|
assert(frontendSettings.enabled, `Frontend extension created with setting 'enabled: false'`);
|
|
this.baseUrl = frontendSettings.base_url;
|
|
this.mqttBaseTopic = settings.get().mqtt.base_topic;
|
|
}
|
|
|
|
override async start(): Promise<void> {
|
|
if (settings.get().frontend.disable_ui_serving) {
|
|
const {host, port} = settings.get().frontend;
|
|
this.wss = new WebSocket.Server({port, host, path: posix.join(this.baseUrl, "api")});
|
|
|
|
logger.info(
|
|
/* v8 ignore next */
|
|
`Frontend UI serving is disabled. WebSocket at: ${this.wss.options.host ?? "0.0.0.0"}:${this.wss.options.port}${this.wss.options.path}`,
|
|
);
|
|
} else {
|
|
const {host, port, ssl_key: sslKey, ssl_cert: sslCert} = settings.get().frontend;
|
|
const hasSSL = (val: string | undefined, key: string): val is string => {
|
|
if (val) {
|
|
if (existsSync(val)) {
|
|
return true;
|
|
}
|
|
|
|
logger.error(`Defined ${key} '${val}' file path does not exists, server won't be secured.`);
|
|
}
|
|
|
|
return false;
|
|
};
|
|
const options: expressStaticGzip.ExpressStaticGzipOptions = {
|
|
enableBrotli: true,
|
|
serveStatic: {
|
|
/* v8 ignore start */
|
|
setHeaders: (res: ServerResponse, path: string): void => {
|
|
if (path.endsWith("index.html")) {
|
|
res.setHeader("Cache-Control", "no-store");
|
|
}
|
|
},
|
|
/* v8 ignore stop */
|
|
},
|
|
};
|
|
const frontend = (await import(settings.get().frontend.package)) as typeof import("zigbee2mqtt-frontend");
|
|
const fileServer = expressStaticGzip(frontend.default.getPath(), options);
|
|
const deviceIconsFileServer = expressStaticGzip(data.joinPath("device_icons"), options);
|
|
const onRequest = (request: IncomingMessage, response: ServerResponse): void => {
|
|
const next = finalhandler(request, response);
|
|
// biome-ignore lint/style/noNonNullAssertion: `Only valid for request obtained from Server`
|
|
const newUrl = posix.relative(this.baseUrl, request.url!);
|
|
|
|
// The request url is not within the frontend base url, so the relative path starts with '..'
|
|
if (newUrl.startsWith(".")) {
|
|
next();
|
|
|
|
return;
|
|
}
|
|
|
|
// Attach originalUrl so that static-server can perform a redirect to '/' when serving the root directory.
|
|
// This is necessary for the browser to resolve relative assets paths correctly.
|
|
request.originalUrl = request.url;
|
|
request.url = `/${newUrl}`;
|
|
request.path = request.url;
|
|
|
|
if (newUrl.startsWith("device_icons/")) {
|
|
request.path = request.path.replace("device_icons/", "");
|
|
request.url = request.url.replace("/device_icons", "");
|
|
|
|
deviceIconsFileServer(request, response, next);
|
|
} else {
|
|
fileServer(request, response, next);
|
|
}
|
|
};
|
|
|
|
if (hasSSL(sslKey, "ssl_key") && hasSSL(sslCert, "ssl_cert")) {
|
|
const serverOptions = {key: readFileSync(sslKey), cert: readFileSync(sslCert)};
|
|
this.server = createSecureServer(serverOptions, onRequest);
|
|
} else {
|
|
this.server = createServer(onRequest);
|
|
}
|
|
|
|
this.server.on("upgrade", this.onUpgrade);
|
|
|
|
if (!host) {
|
|
this.server.listen(port);
|
|
logger.info(`Started frontend on port ${port}`);
|
|
} else if (host.startsWith("/")) {
|
|
this.server.listen(host);
|
|
logger.info(`Started frontend on socket ${host}`);
|
|
} else {
|
|
this.server.listen(port, host);
|
|
logger.info(`Started frontend on port ${host}:${port}`);
|
|
}
|
|
|
|
this.wss = new WebSocket.Server({noServer: true, path: posix.join(this.baseUrl, "api")});
|
|
}
|
|
|
|
this.wss.on("connection", this.onWebSocketConnection);
|
|
|
|
this.eventBus.onMQTTMessagePublished(this, this.onMQTTPublishMessageOrEntityState);
|
|
this.eventBus.onPublishEntityState(this, this.onMQTTPublishMessageOrEntityState);
|
|
}
|
|
|
|
override async stop(): Promise<void> {
|
|
await super.stop();
|
|
|
|
if (this.wss) {
|
|
for (const client of this.wss.clients) {
|
|
client.send(stringify({topic: "bridge/state", payload: {state: "offline"}}));
|
|
client.terminate();
|
|
}
|
|
|
|
this.wss.close();
|
|
}
|
|
|
|
await new Promise((resolve) => (this.server ? this.server.close(resolve) : resolve(undefined)));
|
|
}
|
|
|
|
@bind private onUpgrade(request: IncomingMessage, socket: Socket, head: Buffer): void {
|
|
this.wss.handleUpgrade(request, socket, head, (ws) => {
|
|
// biome-ignore lint/style/noNonNullAssertion: `Only valid for request obtained from Server`
|
|
const {searchParams} = new URL(request.url!, "http://localhost"); // dummy base, may not be absolute
|
|
const authToken = settings.get().frontend.auth_token;
|
|
|
|
if (!authToken || authToken === searchParams.get("token")) {
|
|
this.wss.emit("connection", ws, request);
|
|
} else {
|
|
ws.close(4401, "Unauthorized");
|
|
}
|
|
});
|
|
}
|
|
|
|
@bind private onWebSocketConnection(ws: WebSocket): void {
|
|
ws.on("error", (msg) => logger.error(`WebSocket error: ${msg.message}`));
|
|
ws.on("message", (data: Buffer, isBinary: boolean) => {
|
|
if (!isBinary && data) {
|
|
const message = data.toString();
|
|
const {topic, payload} = JSON.parse(message);
|
|
this.mqtt.onMessage(`${this.mqttBaseTopic}/${topic}`, Buffer.from(stringify(payload)));
|
|
}
|
|
});
|
|
|
|
for (const [topic, payload] of Object.entries(this.mqtt.retainedMessages)) {
|
|
if (topic.startsWith(`${this.mqttBaseTopic}/`)) {
|
|
ws.send(
|
|
stringify({
|
|
// Send topic without base_topic
|
|
topic: topic.substring(this.mqttBaseTopic.length + 1),
|
|
payload: utils.parseJSON(payload.payload, payload.payload),
|
|
}),
|
|
);
|
|
}
|
|
}
|
|
|
|
for (const device of this.zigbee.devicesIterator(utils.deviceNotCoordinator)) {
|
|
const payload = this.state.get(device);
|
|
const lastSeen = settings.get().advanced.last_seen;
|
|
|
|
if (lastSeen !== "disable") {
|
|
payload.last_seen = utils.formatDate(device.zh.lastSeen ?? /* v8 ignore next */ 0, lastSeen);
|
|
}
|
|
|
|
if (device.zh.linkquality !== undefined) {
|
|
payload.linkquality = device.zh.linkquality;
|
|
}
|
|
|
|
ws.send(stringify({topic: device.name, payload}));
|
|
}
|
|
}
|
|
|
|
@bind private onMQTTPublishMessageOrEntityState(data: eventdata.MQTTMessagePublished | eventdata.PublishEntityState): void {
|
|
let topic: string;
|
|
let payload: KeyValue | string;
|
|
|
|
if ("topic" in data) {
|
|
// MQTTMessagePublished
|
|
if (data.options.meta.isEntityState || !data.topic.startsWith(`${this.mqttBaseTopic}/`)) {
|
|
// Don't send entity state to frontend on `MQTTMessagePublished` event, this is handled by
|
|
// `PublishEntityState` instead. Reason for this is to skip attribute messages when `output` is
|
|
// set to `attribute` or `attribute_and_json`, we only want to send JSON entity states to the
|
|
// frontend.
|
|
return;
|
|
}
|
|
// Send topic without base_topic
|
|
topic = data.topic.substring(this.mqttBaseTopic.length + 1);
|
|
payload = utils.parseJSON(data.payload, data.payload);
|
|
} else {
|
|
// PublishEntityState
|
|
topic = data.entity.name;
|
|
payload = data.message;
|
|
}
|
|
|
|
for (const client of this.wss.clients) {
|
|
if (client.readyState === WebSocket.OPEN) {
|
|
client.send(stringify({topic, payload}));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
export default Frontend;
|