mirror of
				https://github.com/louislam/uptime-kuma.git
				synced 2025-10-25 15:59:20 +08:00 
			
		
		
		
	add graceful shutdown
This commit is contained in:
		
							
								
								
									
										11
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										11
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							| @@ -1,7 +1,8 @@ | ||||
| { | ||||
|   "name": "uptime-kuma", | ||||
|   "requires": true, | ||||
|   "version": "1.0.4", | ||||
|   "lockfileVersion": 1, | ||||
|   "requires": true, | ||||
|   "dependencies": { | ||||
|     "@babel/helper-validator-identifier": { | ||||
|       "version": "7.14.5", | ||||
| @@ -1518,6 +1519,14 @@ | ||||
|         "toidentifier": "1.0.0" | ||||
|       } | ||||
|     }, | ||||
|     "http-graceful-shutdown": { | ||||
|       "version": "3.1.2", | ||||
|       "resolved": "https://registry.npmjs.org/http-graceful-shutdown/-/http-graceful-shutdown-3.1.2.tgz", | ||||
|       "integrity": "sha512-2vmU3kWOsZqZy4Kn4EZp00CF+6glpNNN/NAYJPkO9bnMX/D8sRl29TsxIu9Vgyo8ygtCWazWJp720zHfqhSdXg==", | ||||
|       "requires": { | ||||
|         "debug": "^4.3.1" | ||||
|       } | ||||
|     }, | ||||
|     "http-signature": { | ||||
|       "version": "1.2.0", | ||||
|       "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", | ||||
|   | ||||
| @@ -21,6 +21,7 @@ | ||||
|         "dayjs": "^1.10.4", | ||||
|         "express": "^4.17.1", | ||||
|         "form-data": "^4.0.0", | ||||
|         "http-graceful-shutdown": "^3.1.2", | ||||
|         "jsonwebtoken": "^8.5.1", | ||||
|         "nodemailer": "^6.6.2", | ||||
|         "password-hash": "^1.2.2", | ||||
|   | ||||
| @@ -1,9 +1,8 @@ | ||||
| console.log("Welcome to Uptime Kuma ") | ||||
| console.log("Importing libraries") | ||||
| const express = require('express'); | ||||
| const app = express(); | ||||
| const http = require('http'); | ||||
| const server = http.createServer(app); | ||||
| const { Server } = require("socket.io"); | ||||
| const io = new Server(server); | ||||
| const dayjs = require("dayjs"); | ||||
| const {R} = require("redbean-node"); | ||||
| const passwordHash = require('./password-hash'); | ||||
| @@ -12,12 +11,20 @@ const Monitor = require("./model/monitor"); | ||||
| const fs = require("fs"); | ||||
| const {getSettings} = require("./util-server"); | ||||
| const {Notification} = require("./notification") | ||||
| const gracefulShutdown = require('http-graceful-shutdown'); | ||||
| const {sleep} = require("./util"); | ||||
| const args = require('args-parser')(process.argv); | ||||
|  | ||||
| const version = require('../package.json').version; | ||||
| const hostname = args.host || "0.0.0.0" | ||||
| const port = args.port || 3001 | ||||
|  | ||||
| console.log("Version: " + version) | ||||
|  | ||||
| console.log("Creating express and socket.io instance") | ||||
| const app = express(); | ||||
| const server = http.createServer(app); | ||||
| const io = new Server(server); | ||||
| app.use(express.json()) | ||||
|  | ||||
| let totalClient = 0; | ||||
| @@ -539,11 +546,11 @@ async function initDatabase() { | ||||
|     const path = './data/kuma.db'; | ||||
|  | ||||
|     if (! fs.existsSync(path)) { | ||||
|         console.log("Copy Database") | ||||
|         console.log("Copying Database") | ||||
|         fs.copyFileSync("./db/kuma.db", path); | ||||
|     } | ||||
|  | ||||
|     console.log("Connect to Database") | ||||
|     console.log("Connecting to Database") | ||||
|  | ||||
|     R.setup('sqlite', { | ||||
|         filename: path | ||||
| @@ -660,3 +667,72 @@ async function sendImportantHeartbeatList(socket, monitorID) { | ||||
|  | ||||
|     socket.emit("importantHeartbeatList", monitorID, list) | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| const startGracefulShutdown = async () => { | ||||
|     console.log('Shutdown requested'); | ||||
|  | ||||
|  | ||||
|     await (new Promise((resolve) => { | ||||
|         server.close(async function () { | ||||
|             console.log('Stopped Express.'); | ||||
|             process.exit(0) | ||||
|             setTimeout(async () =>{ | ||||
|                 await R.close(); | ||||
|                 console.log("Stopped DB") | ||||
|  | ||||
|                 resolve(); | ||||
|             }, 5000) | ||||
|  | ||||
|         }); | ||||
|     })); | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
| let noReject = true; | ||||
| process.on('unhandledRejection', (reason, p) => { | ||||
|     noReject = false; | ||||
| }); | ||||
|  | ||||
| async function shutdownFunction(signal) { | ||||
|     console.log('Called signal: ' + signal); | ||||
|  | ||||
|     console.log("Stopping all monitors") | ||||
|     for (let id in monitorList) { | ||||
|         let monitor = monitorList[id] | ||||
|         monitor.stop() | ||||
|     } | ||||
|     await sleep(2000) | ||||
|  | ||||
|     console.log("Closing DB") | ||||
|  | ||||
|     // Special handle, because tarn.js throw a promise reject that cannot be caught | ||||
|     while (true) { | ||||
|         noReject = true; | ||||
|         await R.close() | ||||
|         await sleep(2000) | ||||
|  | ||||
|         if (noReject) { | ||||
|             break; | ||||
|         } else { | ||||
|             console.log("Waiting...") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     console.log("OK") | ||||
| } | ||||
|  | ||||
| function finalFunction() { | ||||
|     console.log('Graceful Shutdown') | ||||
| } | ||||
|  | ||||
| gracefulShutdown(server, { | ||||
|     signals: 'SIGINT SIGTERM', | ||||
|     timeout: 30000,                   // timeout: 30 secs | ||||
|     development: false,               // not in dev mode | ||||
|     forceExit: true,                  // triggers process.exit() at the end of shutdown process | ||||
|     onShutdown: shutdownFunction,     // shutdown function (async) - e.g. for cleanup DB, ... | ||||
|     finally: finalFunction            // finally function (sync) - e.g. for logging | ||||
| }); | ||||
|   | ||||
| @@ -5,11 +5,11 @@ | ||||
|  | ||||
|  | ||||
|  | ||||
| export function sleep(ms) { | ||||
| exports.sleep = function (ms) { | ||||
|     return new Promise(resolve => setTimeout(resolve, ms)); | ||||
| } | ||||
|  | ||||
| export function ucfirst(str) { | ||||
| exports.ucfirst = function (str) { | ||||
|     if (! str) { | ||||
|         return str; | ||||
|     } | ||||
|   | ||||
| @@ -5,7 +5,7 @@ | ||||
|  | ||||
| <script> | ||||
|  | ||||
| import {sleep} from "../../server/util"; | ||||
| const {sleep} = require("../../server/util") | ||||
|  | ||||
| export default { | ||||
|  | ||||
|   | ||||
| @@ -220,7 +220,7 @@ | ||||
|  | ||||
| <script> | ||||
| import { Modal } from 'bootstrap' | ||||
| import { ucfirst } from "../../server/util"; | ||||
| const {ucfirst} = require("../../server/util") | ||||
| import axios from "axios"; | ||||
| import { useToast } from 'vue-toastification' | ||||
| import Confirm from "./Confirm.vue"; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user