mirror of
				https://github.com/louislam/uptime-kuma.git
				synced 2025-10-27 00:59:23 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			272 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			272 lines
		
	
	
		
			9.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const express = require("express");
 | |
| const { log } = require("../src/util");
 | |
| const expressStaticGzip = require("express-static-gzip");
 | |
| const fs = require("fs");
 | |
| const path = require("path");
 | |
| const Database = require("./database");
 | |
| const { allowDevAllOrigin } = require("./util-server");
 | |
| const mysql = require("mysql2/promise");
 | |
| 
 | |
| /**
 | |
|  *  A standalone express app that is used to setup a database
 | |
|  *  It is used when db-config.json and kuma.db are not found or invalid
 | |
|  *  Once it is configured, it will shut down and start the main server
 | |
|  */
 | |
| class SetupDatabase {
 | |
|     /**
 | |
|      * Show Setup Page
 | |
|      * @type {boolean}
 | |
|      */
 | |
|     needSetup = true;
 | |
|     /**
 | |
|      * If the server has finished the setup
 | |
|      * @type {boolean}
 | |
|      * @private
 | |
|      */
 | |
|     runningSetup = false;
 | |
|     /**
 | |
|      * @inheritDoc
 | |
|      * @type {UptimeKumaServer}
 | |
|      * @private
 | |
|      */
 | |
|     server;
 | |
| 
 | |
|     /**
 | |
|      * @param  {object} args The arguments passed from the command line
 | |
|      * @param  {UptimeKumaServer} server the main server instance
 | |
|      */
 | |
|     constructor(args, server) {
 | |
|         this.server = server;
 | |
| 
 | |
|         // Priority: env > db-config.json
 | |
|         // If env is provided, write it to db-config.json
 | |
|         // If db-config.json is found, check if it is valid
 | |
|         // If db-config.json is not found or invalid, check if kuma.db is found
 | |
|         // If kuma.db is not found, show setup page
 | |
| 
 | |
|         let dbConfig;
 | |
| 
 | |
|         try {
 | |
|             dbConfig = Database.readDBConfig();
 | |
|             log.debug("setup-database", "db-config.json is found and is valid");
 | |
|             this.needSetup = false;
 | |
| 
 | |
|         } catch (e) {
 | |
|             log.info("setup-database", "db-config.json is not found or invalid: " + e.message);
 | |
| 
 | |
|             // Check if kuma.db is found (1.X.X users), generate db-config.json
 | |
|             if (fs.existsSync(path.join(Database.dataDir, "kuma.db"))) {
 | |
|                 this.needSetup = false;
 | |
| 
 | |
|                 log.info("setup-database", "kuma.db is found, generate db-config.json");
 | |
|                 Database.writeDBConfig({
 | |
|                     type: "sqlite",
 | |
|                 });
 | |
|             } else {
 | |
|                 this.needSetup = true;
 | |
|             }
 | |
|             dbConfig = {};
 | |
|         }
 | |
| 
 | |
|         if (process.env.UPTIME_KUMA_DB_TYPE) {
 | |
|             this.needSetup = false;
 | |
|             log.info("setup-database", "UPTIME_KUMA_DB_TYPE is provided by env, try to override db-config.json");
 | |
|             dbConfig.type = process.env.UPTIME_KUMA_DB_TYPE;
 | |
|             dbConfig.hostname = process.env.UPTIME_KUMA_DB_HOSTNAME;
 | |
|             dbConfig.port = process.env.UPTIME_KUMA_DB_PORT;
 | |
|             dbConfig.dbName = process.env.UPTIME_KUMA_DB_NAME;
 | |
|             dbConfig.username = process.env.UPTIME_KUMA_DB_USERNAME;
 | |
|             dbConfig.password = process.env.UPTIME_KUMA_DB_PASSWORD;
 | |
|             Database.writeDBConfig(dbConfig);
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Show Setup Page
 | |
|      * @returns {boolean} true if the setup page should be shown
 | |
|      */
 | |
|     isNeedSetup() {
 | |
|         return this.needSetup;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Check if the embedded MariaDB is enabled
 | |
|      * @returns {boolean} true if the embedded MariaDB is enabled
 | |
|      */
 | |
|     isEnabledEmbeddedMariaDB() {
 | |
|         return process.env.UPTIME_KUMA_ENABLE_EMBEDDED_MARIADB === "1";
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Start the setup-database server
 | |
|      * @param {string} hostname where the server is listening
 | |
|      * @param {number} port where the server is listening
 | |
|      * @returns {Promise<void>}
 | |
|      */
 | |
|     start(hostname, port) {
 | |
|         return new Promise((resolve) => {
 | |
|             const app = express();
 | |
|             let tempServer;
 | |
|             app.use(express.json());
 | |
| 
 | |
|             // Disable Keep Alive, otherwise the server will not shutdown, as the client will keep the connection alive
 | |
|             app.use(function (req, res, next) {
 | |
|                 res.setHeader("Connection", "close");
 | |
|                 next();
 | |
|             });
 | |
| 
 | |
|             app.get("/", async (request, response) => {
 | |
|                 response.redirect("/setup-database");
 | |
|             });
 | |
| 
 | |
|             app.get("/api/entry-page", async (request, response) => {
 | |
|                 allowDevAllOrigin(response);
 | |
|                 response.json({
 | |
|                     type: "setup-database",
 | |
|                 });
 | |
|             });
 | |
| 
 | |
|             app.get("/setup-database-info", (request, response) => {
 | |
|                 allowDevAllOrigin(response);
 | |
|                 console.log("Request /setup-database-info");
 | |
|                 response.json({
 | |
|                     runningSetup: this.runningSetup,
 | |
|                     needSetup: this.needSetup,
 | |
|                     isEnabledEmbeddedMariaDB: this.isEnabledEmbeddedMariaDB(),
 | |
|                 });
 | |
|             });
 | |
| 
 | |
|             app.post("/setup-database", async (request, response) => {
 | |
|                 allowDevAllOrigin(response);
 | |
| 
 | |
|                 if (this.runningSetup) {
 | |
|                     response.status(400).json("Setup is already running");
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 this.runningSetup = true;
 | |
| 
 | |
|                 let dbConfig = request.body.dbConfig;
 | |
| 
 | |
|                 let supportedDBTypes = [ "mariadb", "sqlite" ];
 | |
| 
 | |
|                 if (this.isEnabledEmbeddedMariaDB()) {
 | |
|                     supportedDBTypes.push("embedded-mariadb");
 | |
|                 }
 | |
| 
 | |
|                 // Validate input
 | |
|                 if (typeof dbConfig !== "object") {
 | |
|                     response.status(400).json("Invalid dbConfig");
 | |
|                     this.runningSetup = false;
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 if (!dbConfig.type) {
 | |
|                     response.status(400).json("Database Type is required");
 | |
|                     this.runningSetup = false;
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 if (!supportedDBTypes.includes(dbConfig.type)) {
 | |
|                     response.status(400).json("Unsupported Database Type");
 | |
|                     this.runningSetup = false;
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 // External MariaDB
 | |
|                 if (dbConfig.type === "mariadb") {
 | |
|                     if (!dbConfig.hostname) {
 | |
|                         response.status(400).json("Hostname is required");
 | |
|                         this.runningSetup = false;
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     if (!dbConfig.port) {
 | |
|                         response.status(400).json("Port is required");
 | |
|                         this.runningSetup = false;
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     if (!dbConfig.dbName) {
 | |
|                         response.status(400).json("Database name is required");
 | |
|                         this.runningSetup = false;
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     if (!dbConfig.username) {
 | |
|                         response.status(400).json("Username is required");
 | |
|                         this.runningSetup = false;
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     if (!dbConfig.password) {
 | |
|                         response.status(400).json("Password is required");
 | |
|                         this.runningSetup = false;
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     // Test connection
 | |
|                     try {
 | |
|                         const connection = await mysql.createConnection({
 | |
|                             host: dbConfig.hostname,
 | |
|                             port: dbConfig.port,
 | |
|                             user: dbConfig.username,
 | |
|                             password: dbConfig.password,
 | |
|                         });
 | |
|                         await connection.execute("SELECT 1");
 | |
|                         connection.end();
 | |
|                     } catch (e) {
 | |
|                         response.status(400).json("Cannot connect to the database: " + e.message);
 | |
|                         this.runningSetup = false;
 | |
|                         return;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 // Write db-config.json
 | |
|                 Database.writeDBConfig(dbConfig);
 | |
| 
 | |
|                 response.json({
 | |
|                     ok: true,
 | |
|                 });
 | |
| 
 | |
|                 // Shutdown down this express and start the main server
 | |
|                 log.info("setup-database", "Database is configured, close the setup-database server and start the main server now.");
 | |
|                 if (tempServer) {
 | |
|                     tempServer.close(() => {
 | |
|                         log.info("setup-database", "The setup-database server is closed");
 | |
|                         resolve();
 | |
|                     });
 | |
|                 } else {
 | |
|                     resolve();
 | |
|                 }
 | |
| 
 | |
|             });
 | |
| 
 | |
|             app.use("/", expressStaticGzip("dist", {
 | |
|                 enableBrotli: true,
 | |
|             }));
 | |
| 
 | |
|             app.get("*", async (_request, response) => {
 | |
|                 response.send(this.server.indexHTML);
 | |
|             });
 | |
| 
 | |
|             app.options("*", async (_request, response) => {
 | |
|                 allowDevAllOrigin(response);
 | |
|                 response.end();
 | |
|             });
 | |
| 
 | |
|             tempServer = app.listen(port, hostname, () => {
 | |
|                 log.info("setup-database", `Starting Setup Database on ${port}`);
 | |
|                 let domain = (hostname) ? hostname : "localhost";
 | |
|                 log.info("setup-database", `Open http://${domain}:${port} in your browser`);
 | |
|                 log.info("setup-database", "Waiting for user action...");
 | |
|             });
 | |
|         });
 | |
|     }
 | |
| }
 | |
| 
 | |
| module.exports = {
 | |
|     SetupDatabase,
 | |
| };
 |