/* eslint-disable camelcase */ /*! // Common Util for frontend and backend // // DOT NOT MODIFY util.js! // Need to run "npm run tsc" to compile if there are any changes. // // Backend uses the compiled file util.js // Frontend uses util.ts */ import * as dayjs from "dayjs"; // For loading dayjs plugins, don't remove event though it is not used in this file // eslint-disable-next-line @typescript-eslint/no-unused-vars import * as timezone from "dayjs/plugin/timezone"; // eslint-disable-next-line @typescript-eslint/no-unused-vars import * as utc from "dayjs/plugin/utc"; import * as jsonata from "jsonata"; export const isDev = process.env.NODE_ENV === "development"; export const isNode = typeof process !== "undefined" && process?.versions?.node; export const appName = "Uptime Kuma"; export const DOWN = 0; export const UP = 1; export const PENDING = 2; export const MAINTENANCE = 3; export const STATUS_PAGE_ALL_DOWN = 0; export const STATUS_PAGE_ALL_UP = 1; export const STATUS_PAGE_PARTIAL_DOWN = 2; export const STATUS_PAGE_MAINTENANCE = 3; export const SQL_DATE_FORMAT = "YYYY-MM-DD"; export const SQL_DATETIME_FORMAT = "YYYY-MM-DD HH:mm:ss"; export const SQL_DATETIME_FORMAT_WITHOUT_SECOND = "YYYY-MM-DD HH:mm"; export const MAX_INTERVAL_SECOND = 2073600; // 24 days export const MIN_INTERVAL_SECOND = 20; // 20 seconds // Console colors // https://stackoverflow.com/questions/9781218/how-to-change-node-jss-console-font-color export const CONSOLE_STYLE_Reset = "\x1b[0m"; export const CONSOLE_STYLE_Bright = "\x1b[1m"; export const CONSOLE_STYLE_Dim = "\x1b[2m"; export const CONSOLE_STYLE_Underscore = "\x1b[4m"; export const CONSOLE_STYLE_Blink = "\x1b[5m"; export const CONSOLE_STYLE_Reverse = "\x1b[7m"; export const CONSOLE_STYLE_Hidden = "\x1b[8m"; export const CONSOLE_STYLE_FgBlack = "\x1b[30m"; export const CONSOLE_STYLE_FgRed = "\x1b[31m"; export const CONSOLE_STYLE_FgGreen = "\x1b[32m"; export const CONSOLE_STYLE_FgYellow = "\x1b[33m"; export const CONSOLE_STYLE_FgBlue = "\x1b[34m"; export const CONSOLE_STYLE_FgMagenta = "\x1b[35m"; export const CONSOLE_STYLE_FgCyan = "\x1b[36m"; export const CONSOLE_STYLE_FgWhite = "\x1b[37m"; export const CONSOLE_STYLE_FgGray = "\x1b[90m"; export const CONSOLE_STYLE_FgOrange = "\x1b[38;5;208m"; export const CONSOLE_STYLE_FgLightGreen = "\x1b[38;5;119m"; export const CONSOLE_STYLE_FgLightBlue = "\x1b[38;5;117m"; export const CONSOLE_STYLE_FgViolet = "\x1b[38;5;141m"; export const CONSOLE_STYLE_FgBrown = "\x1b[38;5;130m"; export const CONSOLE_STYLE_FgPink = "\x1b[38;5;219m"; export const CONSOLE_STYLE_BgBlack = "\x1b[40m"; export const CONSOLE_STYLE_BgRed = "\x1b[41m"; export const CONSOLE_STYLE_BgGreen = "\x1b[42m"; export const CONSOLE_STYLE_BgYellow = "\x1b[43m"; export const CONSOLE_STYLE_BgBlue = "\x1b[44m"; export const CONSOLE_STYLE_BgMagenta = "\x1b[45m"; export const CONSOLE_STYLE_BgCyan = "\x1b[46m"; export const CONSOLE_STYLE_BgWhite = "\x1b[47m"; export const CONSOLE_STYLE_BgGray = "\x1b[100m"; const consoleModuleColors = [ CONSOLE_STYLE_FgCyan, CONSOLE_STYLE_FgGreen, CONSOLE_STYLE_FgLightGreen, CONSOLE_STYLE_FgBlue, CONSOLE_STYLE_FgLightBlue, CONSOLE_STYLE_FgMagenta, CONSOLE_STYLE_FgOrange, CONSOLE_STYLE_FgViolet, CONSOLE_STYLE_FgBrown, CONSOLE_STYLE_FgPink, ]; const consoleLevelColors : Record = { "INFO": CONSOLE_STYLE_FgCyan, "WARN": CONSOLE_STYLE_FgYellow, "ERROR": CONSOLE_STYLE_FgRed, "DEBUG": CONSOLE_STYLE_FgGray, }; /** * Flip the status of s * @param s input status: UP or DOWN * @returns {number} UP or DOWN */ export const badgeConstants = { naColor: "#999", defaultUpColor: "#66c20a", defaultWarnColor: "#eed202", defaultDownColor: "#c2290a", defaultPendingColor: "#f8a306", defaultMaintenanceColor: "#1747f5", defaultPingColor: "blue", // as defined by badge-maker / shields.io defaultStyle: "flat", defaultPingValueSuffix: "ms", defaultPingLabelSuffix: "h", defaultUptimeValueSuffix: "%", defaultUptimeLabelSuffix: "h", defaultCertExpValueSuffix: " days", defaultCertExpLabelSuffix: "h", // Values Come From Default Notification Times defaultCertExpireWarnDays: "14", defaultCertExpireDownDays: "7" }; /** Flip the status of s */ export function flipStatus(s: number) { if (s === UP) { return DOWN; } if (s === DOWN) { return UP; } return s; } /** * Delays for specified number of seconds * @param ms Number of milliseconds to sleep for * @returns {Promise} Promise that resolves after ms */ export function sleep(ms: number) { return new Promise(resolve => setTimeout(resolve, ms)); } /** * PHP's ucfirst * @param str string input * @returns {string} string with first letter capitalized */ export function ucfirst(str: string) { if (!str) { return str; } const firstLetter = str.substr(0, 1); return firstLetter.toUpperCase() + str.substr(1); } /** * @deprecated Use log.debug (https://github.com/louislam/uptime-kuma/pull/910) * @param msg Message to write * @returns {void} */ export function debug(msg: unknown) { log.log("", msg, "debug"); } class Logger { /** * UPTIME_KUMA_HIDE_LOG=debug_monitor,info_monitor * * Example: * [ * "debug_monitor", // Hide all logs that level is debug and the module is monitor * "info_monitor", * ] */ hideLog : Record = { info: [], warn: [], error: [], debug: [], }; /** * */ constructor() { if (typeof process !== "undefined" && process.env.UPTIME_KUMA_HIDE_LOG) { const list = process.env.UPTIME_KUMA_HIDE_LOG.split(",").map(v => v.toLowerCase()); for (const pair of list) { // split first "_" only const values = pair.split(/_(.*)/s); if (values.length >= 2) { this.hideLog[values[0]].push(values[1]); } } this.debug("server", "UPTIME_KUMA_HIDE_LOG is set"); this.debug("server", this.hideLog); } } /** * Write a message to the log * @param module The module the log comes from * @param msg Message to write * @param level Log level. One of INFO, WARN, ERROR, DEBUG or can be customized. * @returns {void} */ log(module: string, msg: any, level: string) { if (level === "DEBUG" && !isDev) { return; } if (this.hideLog[level] && this.hideLog[level].includes(module.toLowerCase())) { return; } module = module.toUpperCase(); level = level.toUpperCase(); let now; if (dayjs.tz) { now = dayjs.tz(new Date()).format(); } else { now = dayjs().format(); } const levelColor = consoleLevelColors[level]; const moduleColor = consoleModuleColors[intHash(module, consoleModuleColors.length)]; let timePart: string; let modulePart: string; let levelPart: string; let msgPart: string; if (isNode) { // Add console colors switch (level) { case "DEBUG": timePart = CONSOLE_STYLE_FgGray + now + CONSOLE_STYLE_Reset; break; default: timePart = CONSOLE_STYLE_FgCyan + now + CONSOLE_STYLE_Reset; break; } modulePart = "[" + moduleColor + module + CONSOLE_STYLE_Reset + "]"; levelPart = levelColor + `${level}:` + CONSOLE_STYLE_Reset; switch (level) { case "ERROR": if (typeof msg === "string") { msgPart = CONSOLE_STYLE_FgRed + msg + CONSOLE_STYLE_Reset; } else { msgPart = msg; } break; case "DEBUG": if (typeof msg === "string") { msgPart = CONSOLE_STYLE_FgGray + msg + CONSOLE_STYLE_Reset; } else { msgPart = msg; } break; default: msgPart = msg; break; } } else { // No console colors timePart = now; modulePart = `[${module}]`; levelPart = `${level}:`; msgPart = msg; } // Write to console switch (level) { case "ERROR": console.error(timePart, modulePart, levelPart, msgPart); break; case "WARN": console.warn(timePart, modulePart, levelPart, msgPart); break; case "INFO": console.info(timePart, modulePart, levelPart, msgPart); break; case "DEBUG": if (isDev) { console.debug(timePart, modulePart, levelPart, msgPart); } break; default: console.log(timePart, modulePart, levelPart, msgPart); break; } } /** * Log an INFO message * @param module Module log comes from * @param msg Message to write * @returns {void} */ info(module: string, msg: unknown) { this.log(module, msg, "info"); } /** * Log a WARN message * @param module Module log comes from * @param msg Message to write * @returns {void} */ warn(module: string, msg: unknown) { this.log(module, msg, "warn"); } /** * Log an ERROR message * @param module Module log comes from * @param msg Message to write * @returns {void} */ error(module: string, msg: unknown) { this.log(module, msg, "error"); } /** * Log a DEBUG message * @param module Module log comes from * @param msg Message to write * @returns {void} */ debug(module: string, msg: unknown) { this.log(module, msg, "debug"); } /** * Log an exception as an ERROR * @param module Module log comes from * @param exception The exception to include * @param msg The message to write * @returns {void} */ exception(module: string, exception: unknown, msg: unknown) { let finalMessage = exception; if (msg) { finalMessage = `${msg}: ${exception}`; } this.log(module, finalMessage, "error"); } } export const log = new Logger(); declare global { interface String { replaceAll(str: string, newStr: string): string; } } /** * String.prototype.replaceAll() polyfill * https://gomakethings.com/how-to-replace-a-section-of-a-string-with-another-one-with-vanilla-js/ * @author Chris Ferdinandi * @license MIT * @returns {void} */ export function polyfill() { if (!String.prototype.replaceAll) { String.prototype.replaceAll = function (str: string, newStr: string) { // If a regex pattern if (Object.prototype.toString.call(str).toLowerCase() === "[object regexp]") { return this.replace(str, newStr); } // If a string return this.replace(new RegExp(str, "g"), newStr); }; } } export class TimeLogger { startTime: number; /** * */ constructor() { this.startTime = dayjs().valueOf(); } /** * Output time since start of monitor * @param name Name of monitor * @returns {void} */ print(name: string) { if (isDev && process.env.TIMELOGGER === "1") { console.log(name + ": " + (dayjs().valueOf() - this.startTime) + "ms"); } } } /** * Returns a random number between min (inclusive) and max (exclusive) * @param min minumim value, inclusive * @param max maximum value, exclusive * @returns {number} Random number */ export function getRandomArbitrary(min: number, max: number) { return Math.random() * (max - min) + min; } /** * From: https://stackoverflow.com/questions/1527803/generating-random-whole-numbers-in-javascript-in-a-specific-range * * Returns a random integer between min (inclusive) and max (inclusive). * The value is no lower than min (or the next integer greater than min * if min isn't an integer) and no greater than max (or the next integer * lower than max if max isn't an integer). * Using Math.round() will give you a non-uniform distribution! * @param min minumim value, inclusive * @param max maximum value, exclusive * @returns {number} Random number */ export function getRandomInt(min: number, max: number) { min = Math.ceil(min); max = Math.floor(max); return Math.floor(Math.random() * (max - min + 1)) + min; } /** * Returns either the NodeJS crypto.randomBytes() function or its * browser equivalent implemented via window.crypto.getRandomValues() * @returns {Uint8Array} Random bytes */ const getRandomBytes = ( (typeof window !== "undefined" && window.crypto) // Browsers ? function () { return (numBytes: number) => { const randomBytes = new Uint8Array(numBytes); for (let i = 0; i < numBytes; i += 65536) { window.crypto.getRandomValues(randomBytes.subarray(i, i + Math.min(numBytes - i, 65536))); } return randomBytes; }; } // Node : function () { // eslint-disable-next-line @typescript-eslint/no-var-requires return require("crypto").randomBytes; } )(); /** * Get a random integer suitable for use in cryptography between upper * and lower bounds. * @param min Minimum value of integer * @param max Maximum value of integer * @returns Cryptographically suitable random integer */ export function getCryptoRandomInt(min: number, max: number):number { // synchronous version of: https://github.com/joepie91/node-random-number-csprng const range = max - min; if (range >= Math.pow(2, 32)) { console.log("Warning! Range is too large."); } let tmpRange = range; let bitsNeeded = 0; let bytesNeeded = 0; let mask = 1; while (tmpRange > 0) { if (bitsNeeded % 8 === 0) { bytesNeeded += 1; } bitsNeeded += 1; mask = mask << 1 | 1; tmpRange = tmpRange >>> 1; } const randomBytes = getRandomBytes(bytesNeeded); let randomValue = 0; for (let i = 0; i < bytesNeeded; i++) { randomValue |= randomBytes[i] << 8 * i; } randomValue = randomValue & mask; if (randomValue <= range) { return min + randomValue; } else { return getCryptoRandomInt(min, max); } } /** * Generate a random alphanumeric string of fixed length * @param length Length of string to generate * @returns string */ export function genSecret(length = 64) { let secret = ""; const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"; const charsLength = chars.length; for ( let i = 0; i < length; i++ ) { secret += chars.charAt(getCryptoRandomInt(0, charsLength - 1)); } return secret; } /** * Get the path of a monitor * @param id ID of monitor * @returns Formatted relative path */ export function getMonitorRelativeURL(id: string) { return "/dashboard/" + id; } /** * Get relative path for maintenance * @param id ID of maintenance * @returns Formatted relative path */ export function getMaintenanceRelativeURL(id: string) { return "/maintenance/" + id; } /** * Parse to Time Object that used in VueDatePicker * @param {string} time E.g. 12:00 * @returns object * @throws {Error} if time string is invalid */ export function parseTimeObject(time: string) { if (!time) { return { hours: 0, minutes: 0, }; } const array = time.split(":"); if (array.length < 2) { throw new Error("parseVueDatePickerTimeFormat: Invalid Time"); } const obj = { hours: parseInt(array[0]), minutes: parseInt(array[1]), seconds: 0, }; if (array.length >= 3) { obj.seconds = parseInt(array[2]); } return obj; } /** * Parse time to string from object {hours: number, minutes: number, seconds?: number} * @param obj object to parse * @returns {string} e.g. 12:00 */ export function parseTimeFromTimeObject(obj : any) { if (!obj) { return obj; } let result = ""; result += obj.hours.toString().padStart(2, "0") + ":" + obj.minutes.toString().padStart(2, "0"); if (obj.seconds) { result += ":" + obj.seconds.toString().padStart(2, "0"); } return result; } /** * Convert ISO date to UTC * @param input Date * @returns ISO Date time */ export function isoToUTCDateTime(input : string) { return dayjs(input).utc().format(SQL_DATETIME_FORMAT); } /** * @param input valid datetime string * @returns {string} ISO DateTime string */ export function utcToISODateTime(input : string) { return dayjs.utc(input).toISOString(); } /** * For SQL_DATETIME_FORMAT * @param input valid datetime string * @param format Format to return * @returns A string date of SQL_DATETIME_FORMAT */ export function utcToLocal(input : string, format = SQL_DATETIME_FORMAT) : string { return dayjs.utc(input).local().format(format); } /** * Convert local datetime to UTC * @param input Local date * @param format Format to return * @returns Date in requested format */ export function localToUTC(input : string, format = SQL_DATETIME_FORMAT) { return dayjs(input).utc().format(format); } /** * Generate a decimal integer number from a string * @param str Input * @param length Default is 10 which means 0 - 9 * @returns {number} output number */ export function intHash(str : string, length = 10) : number { // A simple hashing function (you can use more complex hash functions if needed) let hash = 0; for (let i = 0; i < str.length; i++) { hash += str.charCodeAt(i); } // Normalize the hash to the range [0, 10] return (hash % length + length) % length; // Ensure the result is non-negative } /** * Evaluate a JSON query expression against the provided data. * @param data The data to evaluate the JSON query against. * @param jsonPath The JSON path or custom JSON query expression. * @param jsonPathOperator The operator to use for comparison. * @param expectedValue The expected value to compare against. * @returns An object containing the status and the evaluation result. * @throws Error if the evaluation returns undefined. */ export async function evaluateJsonQuery(data: any, jsonPath: string, jsonPathOperator: string, expectedValue: any): Promise<{ status: boolean; response: any }> { // Attempt to parse data as JSON; if unsuccessful, handle based on data type. let response: any; try { response = JSON.parse(data); } catch { response = typeof data === "number" || typeof data === "object" ? data : data.toString(); } try { // If a JSON path is provided, pre-evaluate the data using it. response = (jsonPath) ? await jsonata(jsonPath).evaluate(response) : response; if (typeof response === "object" || Array.isArray(response) || response instanceof Date || typeof response === "function") { throw new Error(`The post-JSON query evaluated response from the server is of type ${typeof response}, which cannot be directly compared to the expected value`); } // Perform the comparison logic using the chosen operator let jsonQueryExpression; switch (jsonPathOperator) { case ">": case ">=": case "<": case "<=": jsonQueryExpression = `$number($.value) ${jsonPathOperator} $number($.expected)`; break; case "!=": jsonQueryExpression = "$.value != $.expected"; break; case "==": jsonQueryExpression = "$.value = $.expected"; break; case "contains": jsonQueryExpression = "$contains($.value, $.expected)"; break; default: throw new Error(`Invalid condition ${jsonPathOperator}`); } // Evaluate the JSON Query Expression const expression = jsonata(jsonQueryExpression); const status = await expression.evaluate({ value: response.toString(), expected: expectedValue.toString() }); if (response === undefined || status === undefined) { throw new Error("Query evaluation returned undefined. Check query syntax and the structure of the response data"); } return { status, // The evaluation of the json query response // The response from the server or result from initial json-query evaluation }; } catch (err: any) { response = (response.toString().length > 50) ? `${response.substring(0, 50)}… (truncated)` : response.toString();// Truncate long responses to the console throw new Error(`Error evaluating JSON query: ${err.message}. Response from server was: ${response}`); } }