168 lines
5.7 KiB
TypeScript
Raw Permalink Normal View History

///
2025-02-25 09:39:16 +02:00
/// Copyright © 2016-2025 The Thingsboard Authors
///
/// Licensed under the Apache License, Version 2.0 (the "License");
/// you may not use this file except in compliance with the License.
/// You may obtain a copy of the License at
///
/// http://www.apache.org/licenses/LICENSE-2.0
///
/// Unless required by applicable law or agreed to in writing, software
/// distributed under the License is distributed on an "AS IS" BASIS,
/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
/// See the License for the specific language governing permissions and
/// limitations under the License.
///
import express from 'express';
import config from 'config';
import { _logger} from './config/logger';
import path from 'path';
import http, { ServerResponse } from 'http';
import httpProxy from 'http-proxy';
import compression from 'compression';
import historyApiFallback from 'connect-history-api-fallback';
2022-07-07 15:25:23 +03:00
import { Socket } from 'net';
import { RequestHandler } from 'serve-static';
const logger = _logger('main');
let server: http.Server | null;
2022-07-07 15:25:23 +03:00
let connections: Socket[] = [];
2018-10-02 17:42:09 +03:00
(async() => {
try {
logger.info('Starting ThingsBoard Web UI Microservice...');
const bindAddress: string = config.get('server.address');
const bindPort = Number(config.get('server.port'));
2018-10-02 17:42:09 +03:00
const thingsboardEnableProxy: string = config.get('thingsboard.enableProxy');
const thingsboardHost: string = config.get('thingsboard.host');
const thingsboardPort = Number(config.get('thingsboard.port'));
2018-10-02 17:42:09 +03:00
logger.info('Bind address: %s', bindAddress);
logger.info('Bind port: %s', bindPort);
logger.info('ThingsBoard Enable Proxy: %s', thingsboardEnableProxy);
2018-10-02 17:42:09 +03:00
logger.info('ThingsBoard host: %s', thingsboardHost);
logger.info('ThingsBoard port: %s', thingsboardPort);
const useApiProxy = thingsboardEnableProxy === "true";
let webDir = path.join(__dirname, 'web');
2018-10-02 17:42:09 +03:00
if (typeof process.env.WEB_FOLDER !== 'undefined') {
2018-10-02 17:42:09 +03:00
webDir = path.resolve(process.env.WEB_FOLDER);
}
logger.info('Web folder: %s', webDir);
const app = express();
server = http.createServer(app);
let apiProxy: httpProxy;
if (useApiProxy) {
apiProxy = httpProxy.createProxyServer({
target: {
host: thingsboardHost,
port: thingsboardPort
}
});
apiProxy.on('error', (err, req, res) => {
logger.warn('API proxy error: %s', err.message);
if (res instanceof ServerResponse) {
res.writeHead(500);
const error = err as any;
if (error.code && error.code === 'ECONNREFUSED') {
res.end('Unable to connect to ThingsBoard server.');
} else {
res.end('ThingsBoard server connection error: ' + error.code ? error.code : '');
}
}
});
app.all('/api/*', (req, res) => {
2020-02-26 20:43:39 +02:00
logger.debug(req.method + ' ' + req.originalUrl);
apiProxy.web(req, res);
});
2018-10-02 17:42:09 +03:00
app.all('/static/rulenode/*', (req, res) => {
2020-02-26 20:43:39 +02:00
apiProxy.web(req, res);
});
server.on('upgrade', (req, socket, head) => {
apiProxy.ws(req, socket, head);
});
}
2018-10-02 17:42:09 +03:00
app.use(historyApiFallback());
app.use(compression());
2018-10-02 17:42:09 +03:00
const root = path.join(webDir, 'public');
2018-10-02 17:42:09 +03:00
const staticServe = express.static(root);
const indexHtmlFallback: RequestHandler<any> = (req, res, next) => {
res.sendFile(path.join(root, 'index.html'));
};
app.use(staticServe);
const indexHtmlFallbackPaths = [
/^\/resources\/scada-symbols\/(?:system|tenant)\/[^/]+$/
];
app.use(indexHtmlFallbackPaths, indexHtmlFallback);
2018-10-02 17:42:09 +03:00
server.listen(bindPort, bindAddress, () => {
logger.info('==> 🌎 Listening on port %s.', bindPort);
logger.info('Started ThingsBoard Web UI Microservice.');
2022-07-07 15:25:23 +03:00
}).on('error', async (error) => {
logger.error('Failed to start ThingsBoard Web UI Microservice: %s', error.message);
logger.error(error.stack);
2022-07-07 15:25:23 +03:00
await exit(-1);
});
server.on('connection', connection => {
connections.push(connection);
connection.on('close', () => connections = connections.filter(curr => curr !== connection));
2018-10-02 17:42:09 +03:00
});
} catch (e: any) {
2018-10-02 17:42:09 +03:00
logger.error('Failed to start ThingsBoard Web UI Microservice: %s', e.message);
logger.error(e.stack);
2022-07-07 15:25:23 +03:00
await exit(-1);
2018-10-02 17:42:09 +03:00
}
})();
2022-07-07 15:25:23 +03:00
[`SIGINT`, `SIGUSR1`, `SIGUSR2`, `uncaughtException`, `SIGTERM`].forEach((eventType) => {
process.once(eventType, async () => {
2022-07-07 15:25:23 +03:00
logger.info(`${eventType} signal received`);
await exit(0);
});
})
process.on('exit', async (code: number) => {
logger.info(`ThingsBoard Web UI Microservice has been stopped. Exit code: ${code}.`);
2018-10-02 17:42:09 +03:00
});
2022-07-07 15:25:23 +03:00
async function exit(status: number) {
2018-10-02 17:42:09 +03:00
logger.info('Exiting with status: %d ...', status);
if (server) {
logger.info('Stopping HTTP Server...');
2022-07-07 15:25:23 +03:00
connections.forEach(curr => curr.end(() => curr.destroy()));
const _server = server;
2018-10-02 17:42:09 +03:00
server = null;
2022-07-07 15:25:23 +03:00
const serverClosePromise = new Promise<void>(
(resolve, reject) => {
_server.close((err) => {
logger.info('HTTP Server stopped.');
resolve();
});
}
);
await serverClosePromise;
2018-10-02 17:42:09 +03:00
}
2022-07-07 15:25:23 +03:00
process.exit(status);
2018-10-02 17:42:09 +03:00
}