2021-06-25 15:55:49 +02:00
const dayjs = require ( "dayjs" ) ;
2021-06-27 10:10:55 +02:00
const axios = require ( "axios" ) ;
2021-07-30 13:18:26 +02:00
const { Prometheus } = require ( "../prometheus" ) ;
2023-10-02 23:39:17 +02:00
const { log , UP , DOWN , PENDING , MAINTENANCE , flipStatus , MAX _INTERVAL _SECOND , MIN _INTERVAL _SECOND ,
2024-06-06 00:09:53 +02:00
SQL _DATETIME _FORMAT , evaluateJsonQuery
2023-04-09 10:01:27 +02:00
} = require ( "../../src/util" ) ;
2024-10-09 01:43:44 +02:00
const { tcping , ping , checkCertificate , checkStatusCode , getTotalClientInRoom , setting , mssqlQuery , postgresQuery , mysqlQuery , setSetting , httpNtlm , radius , grpcQuery ,
2024-05-19 21:59:57 +02:00
redisPingAsync , kafkaProducerAsync , getOidcTokenClientCredentials , rootCertificatesFingerprints , axiosAbortSignal
2023-01-05 16:58:24 +02:00
} = require ( "../util-server" ) ;
2021-07-30 13:18:26 +02:00
const { R } = require ( "redbean-node" ) ;
const { BeanModel } = require ( "redbean-node/dist/bean-model" ) ;
2021-09-17 08:42:19 +02:00
const { Notification } = require ( "../notification" ) ;
2021-11-04 13:32:16 +02:00
const { Proxy } = require ( "../proxy" ) ;
2021-10-15 18:57:26 +02:00
const { demoMode } = require ( "../config" ) ;
2021-08-12 18:13:46 +02:00
const version = require ( "../../package.json" ) . version ;
2021-10-09 11:04:51 +02:00
const apicache = require ( "../modules/apicache" ) ;
2022-05-06 08:41:34 +02:00
const { UptimeKumaServer } = require ( "../uptime-kuma-server" ) ;
2022-10-04 10:19:56 +02:00
const { DockerHost } = require ( "../docker" ) ;
2023-01-08 10:22:36 +02:00
const Gamedig = require ( "gamedig" ) ;
2023-06-27 09:54:33 +02:00
const jwt = require ( "jsonwebtoken" ) ;
2023-11-18 05:33:34 +02:00
const crypto = require ( "crypto" ) ;
2023-08-31 23:19:21 +02:00
const { UptimeCalculator } = require ( "../uptime-calculator" ) ;
2023-11-29 11:25:33 +02:00
const { CookieJar } = require ( "tough-cookie" ) ;
const { HttpsCookieAgent } = require ( "http-cookie-agent/http" ) ;
2023-12-10 20:01:56 +02:00
const https = require ( "https" ) ;
const http = require ( "http" ) ;
2021-06-27 10:10:55 +02:00
2023-10-15 20:20:38 +02:00
const rootCertificates = rootCertificatesFingerprints ( ) ;
2021-06-27 10:10:55 +02:00
/ * *
* status :
* 0 = DOWN
* 1 = UP
2021-07-27 19:53:59 +02:00
* 2 = PENDING
2022-01-23 16:22:00 +02:00
* 3 = MAINTENANCE
2021-06-27 10:10:55 +02:00
* /
2021-06-25 15:55:49 +02:00
class Monitor extends BeanModel {
2021-09-12 20:26:45 +02:00
/ * *
2023-08-11 09:46:41 +02:00
* Return an object that ready to parse to JSON for public Only show
* necessary data to public
* @ param { boolean } showTags Include tags in JSON
* @ param { boolean } certExpiry Include certificate expiry info in
* JSON
2024-03-15 16:02:55 +02:00
* @ returns { Promise < object > } Object ready to parse
2021-09-12 20:26:45 +02:00
* /
2023-07-05 01:37:45 +02:00
async toPublicJSON ( showTags = false , certExpiry = false ) {
2022-03-18 11:56:46 +02:00
let obj = {
2021-09-19 13:04:51 +02:00
id : this . id ,
name : this . name ,
2022-06-11 18:23:12 +02:00
sendUrl : this . sendUrl ,
2023-07-17 08:01:10 +02:00
type : this . type ,
2021-09-19 13:04:51 +02:00
} ;
2022-06-11 18:23:12 +02:00
if ( this . sendUrl ) {
obj . url = this . url ;
}
2022-03-18 11:56:46 +02:00
if ( showTags ) {
obj . tags = await this . getTags ( ) ;
}
2023-07-05 01:37:45 +02:00
2023-09-23 21:18:18 +02:00
if ( certExpiry && ( this . type === "http" || this . type === "keyword" || this . type === "json-query" ) && this . getURLProtocol ( ) === "https:" ) {
2023-07-05 01:37:45 +02:00
const { certExpiryDaysRemaining , validCert } = await this . getCertExpiry ( this . id ) ;
obj . certExpiryDaysRemaining = certExpiryDaysRemaining ;
obj . validCert = validCert ;
}
2022-03-18 11:56:46 +02:00
return obj ;
2021-09-12 20:26:45 +02:00
}
/ * *
2022-03-18 11:56:46 +02:00
* Return an object that ready to parse to JSON
2024-10-06 03:36:54 +02:00
* @ param { object } preloadData to prevent n + 1 problems , we query the data in a batch outside of this function
2023-08-11 09:46:41 +02:00
* @ param { boolean } includeSensitiveData Include sensitive data in
* JSON
2024-10-06 03:36:54 +02:00
* @ returns { object } Object ready to parse
2021-09-12 20:26:45 +02:00
* /
2024-10-06 03:36:54 +02:00
toJSON ( preloadData = { } , includeSensitiveData = true ) {
2021-08-26 12:55:19 +02:00
2023-06-27 09:54:33 +02:00
let screenshot = null ;
if ( this . type === "real-browser" ) {
screenshot = "/screenshots/" + jwt . sign ( this . id , UptimeKumaServer . getInstance ( ) . jwtSecret ) + ".png" ;
}
2024-10-06 03:36:54 +02:00
const path = preloadData . paths . get ( this . id ) || [ ] ;
2024-03-03 11:59:38 +02:00
const pathName = path . join ( " / " ) ;
2022-04-17 13:30:58 +02:00
let data = {
2021-06-25 15:55:49 +02:00
id : this . id ,
name : this . name ,
2021-11-12 01:06:32 +02:00
description : this . description ,
2024-03-03 11:59:38 +02:00
path ,
pathName ,
2023-01-28 03:58:03 +02:00
parent : this . parent ,
2024-10-06 03:36:54 +02:00
childrenIDs : preloadData . childrenIDs . get ( this . id ) || [ ] ,
2021-06-25 15:55:49 +02:00
url : this . url ,
2021-10-02 16:48:27 +02:00
method : this . method ,
2021-07-01 08:03:06 +02:00
hostname : this . hostname ,
port : this . port ,
2021-07-19 18:23:06 +02:00
maxretries : this . maxretries ,
2021-07-01 07:11:16 +02:00
weight : this . weight ,
2024-10-06 03:36:54 +02:00
active : preloadData . activeStatus . get ( this . id ) ,
forceInactive : preloadData . forceInactive . get ( this . id ) ,
2021-06-25 15:55:49 +02:00
type : this . type ,
2023-08-06 18:14:56 +02:00
timeout : this . timeout ,
2021-06-25 15:55:49 +02:00
interval : this . interval ,
2021-09-11 18:54:55 +02:00
retryInterval : this . retryInterval ,
2022-01-23 16:22:57 +02:00
resendInterval : this . resendInterval ,
2021-07-01 11:19:28 +02:00
keyword : this . keyword ,
2023-04-06 02:10:21 +02:00
invertKeyword : this . isInvertKeyword ( ) ,
2022-04-05 15:27:50 +02:00
expiryNotification : this . isEnabledExpiryNotification ( ) ,
2021-07-30 16:11:14 +02:00
ignoreTls : this . getIgnoreTls ( ) ,
2021-07-30 18:01:04 +02:00
upsideDown : this . isUpsideDown ( ) ,
2022-07-14 09:32:51 +02:00
packetSize : this . packetSize ,
2021-08-05 13:04:38 +02:00
maxredirects : this . maxredirects ,
accepted _statuscodes : this . getAcceptedStatuscodes ( ) ,
2021-08-23 00:05:48 +02:00
dns _resolve _type : this . dns _resolve _type ,
dns _resolve _server : this . dns _resolve _server ,
2021-08-28 21:20:25 +02:00
dns _last _result : this . dns _last _result ,
2022-01-13 18:17:07 +02:00
docker _container : this . docker _container ,
2022-07-22 17:47:04 +02:00
docker _host : this . docker _host ,
2021-10-30 19:37:15 +02:00
proxyId : this . proxy _id ,
2024-10-06 03:36:54 +02:00
notificationIDList : preloadData . notifications . get ( this . id ) || { } ,
tags : preloadData . tags . get ( this . id ) || [ ] ,
maintenance : preloadData . maintenanceStatus . get ( this . id ) ,
2021-12-18 23:35:18 +02:00
mqttTopic : this . mqttTopic ,
2022-05-13 19:58:23 +02:00
mqttSuccessMessage : this . mqttSuccessMessage ,
2023-12-02 19:36:19 +02:00
mqttCheckType : this . mqttCheckType ,
2022-05-13 02:54:02 +02:00
databaseQuery : this . databaseQuery ,
2022-05-13 19:58:23 +02:00
authMethod : this . authMethod ,
2022-08-03 07:00:39 +02:00
grpcUrl : this . grpcUrl ,
grpcProtobuf : this . grpcProtobuf ,
grpcMethod : this . grpcMethod ,
grpcServiceName : this . grpcServiceName ,
2022-08-03 08:39:31 +02:00
grpcEnableTls : this . getGrpcEnableTls ( ) ,
2022-05-12 11:48:38 +02:00
radiusCalledStationId : this . radiusCalledStationId ,
radiusCallingStationId : this . radiusCallingStationId ,
2023-01-08 10:22:36 +02:00
game : this . game ,
2023-08-07 21:14:21 +02:00
gamedigGivenPortOnly : this . getGameDigGivenPortOnly ( ) ,
2023-06-27 09:54:33 +02:00
httpBodyEncoding : this . httpBodyEncoding ,
2023-07-13 17:37:26 +02:00
jsonPath : this . jsonPath ,
expectedValue : this . expectedValue ,
2023-07-17 10:15:44 +02:00
kafkaProducerTopic : this . kafkaProducerTopic ,
kafkaProducerBrokers : JSON . parse ( this . kafkaProducerBrokers ) ,
2023-10-28 08:42:55 +02:00
kafkaProducerSsl : this . getKafkaProducerSsl ( ) ,
kafkaProducerAllowAutoTopicCreation : this . getKafkaProducerAllowAutoTopicCreation ( ) ,
2023-07-17 10:15:44 +02:00
kafkaProducerMessage : this . kafkaProducerMessage ,
2023-06-27 09:54:33 +02:00
screenshot ,
2024-08-24 22:57:31 +02:00
cacheBust : this . getCacheBust ( ) ,
2023-12-01 09:29:10 +02:00
remote _browser : this . remote _browser ,
2024-04-30 23:13:16 +02:00
snmpOid : this . snmpOid ,
2024-06-05 23:37:47 +02:00
jsonPathOperator : this . jsonPathOperator ,
2024-04-30 23:13:16 +02:00
snmpVersion : this . snmpVersion ,
2024-10-20 15:57:03 +02:00
rabbitmqNodes : JSON . parse ( this . rabbitmqNodes ) ,
2024-08-30 21:48:13 +02:00
conditions : JSON . parse ( this . conditions ) ,
2021-06-25 15:55:49 +02:00
} ;
2022-04-17 13:30:58 +02:00
if ( includeSensitiveData ) {
data = {
... data ,
headers : this . headers ,
body : this . body ,
2022-08-03 07:00:39 +02:00
grpcBody : this . grpcBody ,
grpcMetadata : this . grpcMetadata ,
2022-04-17 13:30:58 +02:00
basic _auth _user : this . basic _auth _user ,
basic _auth _pass : this . basic _auth _pass ,
2023-08-02 09:40:19 +02:00
oauth _client _id : this . oauth _client _id ,
oauth _client _secret : this . oauth _client _secret ,
oauth _token _url : this . oauth _token _url ,
oauth _scopes : this . oauth _scopes ,
oauth _auth _method : this . oauth _auth _method ,
2022-04-17 13:30:58 +02:00
pushToken : this . pushToken ,
2022-12-05 12:18:19 +02:00
databaseConnectionString : this . databaseConnectionString ,
radiusUsername : this . radiusUsername ,
radiusPassword : this . radiusPassword ,
radiusSecret : this . radiusSecret ,
mqttUsername : this . mqttUsername ,
mqttPassword : this . mqttPassword ,
authWorkstation : this . authWorkstation ,
authDomain : this . authDomain ,
2023-01-04 09:37:30 +02:00
tlsCa : this . tlsCa ,
tlsCert : this . tlsCert ,
tlsKey : this . tlsKey ,
2023-07-17 10:15:44 +02:00
kafkaProducerSaslOptions : JSON . parse ( this . kafkaProducerSaslOptions ) ,
2024-10-20 15:57:03 +02:00
rabbitmqUsername : this . rabbitmqUsername ,
rabbitmqPassword : this . rabbitmqPassword ,
2022-04-17 13:30:58 +02:00
} ;
}
2022-12-05 12:18:19 +02:00
data . includeSensitiveData = includeSensitiveData ;
2022-04-17 13:30:58 +02:00
return data ;
2021-06-25 15:55:49 +02:00
}
2022-04-21 19:30:04 +02:00
/ * *
* Get all tags applied to this monitor
2023-08-11 09:46:41 +02:00
* @ returns { Promise < LooseObject < any > [ ] > } List of tags on the
* monitor
2022-04-21 19:30:04 +02:00
* /
2022-03-18 11:56:46 +02:00
async getTags ( ) {
2023-02-15 21:06:29 +02:00
return await R . getAll ( "SELECT mt.*, tag.name, tag.color FROM monitor_tag mt JOIN tag ON mt.tag_id = tag.id WHERE mt.monitor_id = ? ORDER BY tag.name" , [ this . id ] ) ;
2022-03-18 11:56:46 +02:00
}
2023-07-05 01:37:45 +02:00
/ * *
* Gets certificate expiry for this monitor
* @ param { number } monitorID ID of monitor to send
2023-08-11 09:46:41 +02:00
* @ returns { Promise < LooseObject < any >> } Certificate expiry info for
* monitor
2023-07-05 01:37:45 +02:00
* /
async getCertExpiry ( monitorID ) {
let tlsInfoBean = await R . findOne ( "monitor_tls_info" , "monitor_id = ?" , [
monitorID ,
] ) ;
let tlsInfo ;
if ( tlsInfoBean ) {
tlsInfo = JSON . parse ( tlsInfoBean ? . info _json ) ;
if ( tlsInfo ? . valid && tlsInfo ? . certInfo ? . daysRemaining ) {
return {
certExpiryDaysRemaining : tlsInfo . certInfo . daysRemaining ,
validCert : true
} ;
}
}
return {
2023-07-14 02:00:23 +02:00
certExpiryDaysRemaining : "" ,
2023-07-05 01:37:45 +02:00
validCert : false
} ;
}
2021-11-02 13:30:44 +02:00
/ * *
* Encode user and password to Base64 encoding
* for HTTP "basic" auth , as per RFC - 7617
2024-01-06 20:06:06 +02:00
* @ param { string | null } user - The username ( nullable if not changed by a user )
* @ param { string | null } pass - The password ( nullable if not changed by a user )
2024-04-26 09:01:03 +02:00
* @ returns { string } Encoded Base64 string
2021-11-02 13:30:44 +02:00
* /
2021-11-23 06:59:48 +02:00
encodeBase64 ( user , pass ) {
2024-01-06 20:06:06 +02:00
return Buffer . from ( ` ${ user || "" } : ${ pass || "" } ` ) . toString ( "base64" ) ;
2021-11-02 14:11:33 +02:00
}
2021-11-02 13:30:44 +02:00
2022-04-21 19:30:04 +02:00
/ * *
* Is the TLS expiry notification enabled ?
2023-08-11 09:46:41 +02:00
* @ returns { boolean } Enabled ?
2022-04-21 19:30:04 +02:00
* /
2022-04-05 15:27:50 +02:00
isEnabledExpiryNotification ( ) {
return Boolean ( this . expiryNotification ) ;
}
2021-07-30 16:11:14 +02:00
/ * *
* Parse to boolean
2023-08-11 09:46:41 +02:00
* @ returns { boolean } Should TLS errors be ignored ?
2021-07-30 16:11:14 +02:00
* /
getIgnoreTls ( ) {
2021-09-17 08:42:19 +02:00
return Boolean ( this . ignoreTls ) ;
2021-07-30 16:11:14 +02:00
}
/ * *
* Parse to boolean
2023-08-11 09:46:41 +02:00
* @ returns { boolean } Is the monitor in upside down mode ?
2021-07-30 16:11:14 +02:00
* /
2021-07-30 18:01:04 +02:00
isUpsideDown ( ) {
2021-07-30 16:11:14 +02:00
return Boolean ( this . upsideDown ) ;
}
2023-04-06 02:10:21 +02:00
/ * *
* Parse to boolean
2023-08-11 09:46:41 +02:00
* @ returns { boolean } Invert keyword match ?
2023-04-06 02:10:21 +02:00
* /
isInvertKeyword ( ) {
return Boolean ( this . invertKeyword ) ;
}
2022-08-03 08:39:31 +02:00
/ * *
* Parse to boolean
2023-08-11 09:46:41 +02:00
* @ returns { boolean } Enable TLS for gRPC ?
2022-08-03 08:39:31 +02:00
* /
getGrpcEnableTls ( ) {
return Boolean ( this . grpcEnableTls ) ;
}
2024-08-24 22:57:31 +02:00
/ * *
* Parse to boolean
* @ returns { boolean } if cachebusting is enabled
* /
getCacheBust ( ) {
return Boolean ( this . cacheBust ) ;
}
2022-04-16 22:11:45 +02:00
/ * *
2022-04-22 19:42:47 +02:00
* Get accepted status codes
2023-08-11 09:46:41 +02:00
* @ returns { object } Accepted status codes
2022-04-16 22:11:45 +02:00
* /
2021-08-05 13:04:38 +02:00
getAcceptedStatuscodes ( ) {
return JSON . parse ( this . accepted _statuscodes _json ) ;
}
2023-08-11 16:29:45 +02:00
/ * *
2023-09-07 09:42:44 +02:00
* Get if game dig should only use the port which was provided
* @ returns { boolean } gamedig should only use the provided port
2023-08-11 16:29:45 +02:00
* /
2023-08-07 21:14:21 +02:00
getGameDigGivenPortOnly ( ) {
return Boolean ( this . gamedigGivenPortOnly ) ;
}
2023-10-28 08:42:55 +02:00
/ * *
* Parse to boolean
* @ returns { boolean } Kafka Producer Ssl enabled ?
* /
getKafkaProducerSsl ( ) {
return Boolean ( this . kafkaProducerSsl ) ;
}
/ * *
* Parse to boolean
* @ returns { boolean } Kafka Producer Allow Auto Topic Creation Enabled ?
* /
getKafkaProducerAllowAutoTopicCreation ( ) {
return Boolean ( this . kafkaProducerAllowAutoTopicCreation ) ;
}
2022-04-16 22:11:45 +02:00
/ * *
* Start monitor
* @ param { Server } io Socket server instance
2024-01-30 18:54:04 +02:00
* @ returns { Promise < void > }
2022-04-16 22:11:45 +02:00
* /
2023-02-23 18:16:49 +02:00
async start ( io ) {
2021-06-29 10:06:20 +02:00
let previousBeat = null ;
2021-07-19 18:23:06 +02:00
let retries = 0 ;
2021-06-29 10:06:20 +02:00
2024-10-09 01:17:11 +02:00
this . prometheus = new Prometheus ( this ) ;
2021-07-22 17:00:11 +02:00
2021-06-27 10:10:55 +02:00
const beat = async ( ) => {
2021-09-08 13:54:37 +02:00
2022-02-24 09:11:17 +02:00
let beatInterval = this . interval ;
if ( ! beatInterval ) {
beatInterval = 1 ;
}
if ( demoMode ) {
if ( beatInterval < 20 ) {
console . log ( "beat interval too low, reset to 20s" ) ;
beatInterval = 20 ;
}
}
2021-08-10 11:51:30 +02:00
// Expose here for prometheus update
// undefined if not https
let tlsInfo = undefined ;
2022-05-09 21:10:12 +02:00
if ( ! previousBeat || this . type === "push" ) {
2021-06-29 10:06:20 +02:00
previousBeat = await R . findOne ( "heartbeat" , " monitor_id = ? ORDER BY time DESC" , [
2021-07-30 13:18:26 +02:00
this . id ,
2021-09-17 08:42:19 +02:00
] ) ;
2023-11-24 12:11:36 +02:00
if ( previousBeat ) {
retries = previousBeat . retries ;
}
2021-06-29 10:06:20 +02:00
}
2021-07-24 05:42:14 +02:00
const isFirstBeat = ! previousBeat ;
2021-09-17 08:42:19 +02:00
let bean = R . dispense ( "heartbeat" ) ;
2021-06-27 10:10:55 +02:00
bean . monitor _id = this . id ;
2022-03-28 23:28:50 +02:00
bean . time = R . isoDateTimeMillis ( dayjs . utc ( ) ) ;
2021-07-24 05:42:14 +02:00
bean . status = DOWN ;
2022-06-15 16:56:26 +02:00
bean . downCount = previousBeat ? . downCount || 0 ;
2021-06-27 10:10:55 +02:00
2021-07-30 18:01:04 +02:00
if ( this . isUpsideDown ( ) ) {
bean . status = flipStatus ( bean . status ) ;
}
2023-11-12 07:50:51 +02:00
// Runtime patch timeout if it is 0
// See https://github.com/louislam/uptime-kuma/pull/3961#issuecomment-1804149144
2023-11-17 19:17:54 +02:00
if ( ! this . timeout || this . timeout <= 0 ) {
2023-11-12 07:50:51 +02:00
this . timeout = this . interval * 1000 * 0.8 ;
}
2021-06-27 10:10:55 +02:00
try {
2022-01-25 20:07:27 +02:00
if ( await Monitor . isUnderMaintenance ( this . id ) ) {
2022-01-23 16:22:00 +02:00
bean . msg = "Monitor under maintenance" ;
bean . status = MAINTENANCE ;
2023-01-28 03:58:03 +02:00
} else if ( this . type === "group" ) {
const children = await Monitor . getChildren ( this . id ) ;
2023-01-28 14:39:17 +02:00
if ( children . length > 0 ) {
bean . status = UP ;
2023-02-20 14:48:16 +02:00
bean . msg = "All children up and running" ;
2023-01-28 14:39:17 +02:00
for ( const child of children ) {
if ( ! child . active ) {
// Ignore inactive childs
continue ;
}
const lastBeat = await Monitor . getPreviousHeartbeat ( child . id ) ;
// Only change state if the monitor is in worse conditions then the ones before
2023-08-04 08:48:21 +02:00
// lastBeat.status could be null
if ( ! lastBeat ) {
bean . status = PENDING ;
} else if ( bean . status === UP && ( lastBeat . status === PENDING || lastBeat . status === DOWN ) ) {
2023-01-28 14:39:17 +02:00
bean . status = lastBeat . status ;
} else if ( bean . status === PENDING && lastBeat . status === DOWN ) {
bean . status = lastBeat . status ;
}
2023-01-28 03:58:03 +02:00
}
2023-01-28 14:39:17 +02:00
if ( bean . status !== UP ) {
bean . msg = "Child inaccessible" ;
}
} else {
// Set status pending if group is empty
bean . status = PENDING ;
2023-02-01 21:19:47 +02:00
bean . msg = "Group empty" ;
2023-01-28 03:58:03 +02:00
}
2023-01-28 14:39:17 +02:00
2023-07-13 17:37:26 +02:00
} else if ( this . type === "http" || this . type === "keyword" || this . type === "json-query" ) {
2021-08-23 12:52:24 +02:00
// Do not do any queries/high loading things before the "bean.ping"
2021-06-27 10:10:55 +02:00
let startTime = dayjs ( ) . valueOf ( ) ;
2021-07-30 16:11:14 +02:00
2021-11-02 13:30:44 +02:00
// HTTP basic auth
2021-11-04 11:12:06 +02:00
let basicAuthHeader = { } ;
2022-05-13 19:58:23 +02:00
if ( this . auth _method === "basic" ) {
2021-11-04 11:12:06 +02:00
basicAuthHeader = {
2021-11-23 06:59:48 +02:00
"Authorization" : "Basic " + this . encodeBase64 ( this . basic _auth _user , this . basic _auth _pass ) ,
2021-11-02 14:11:33 +02:00
} ;
2021-11-02 13:30:44 +02:00
}
2023-08-02 09:40:19 +02:00
// OIDC: Basic client credential flow.
// Additional grants might be implemented in the future
let oauth2AuthHeader = { } ;
if ( this . auth _method === "oauth2-cc" ) {
try {
if ( this . oauthAccessToken === undefined || new Date ( this . oauthAccessToken . expires _at * 1000 ) <= new Date ( ) ) {
2023-12-17 11:21:07 +02:00
this . oauthAccessToken = await this . makeOidcTokenClientCredentialsRequest ( ) ;
2023-08-02 09:40:19 +02:00
}
oauth2AuthHeader = {
"Authorization" : this . oauthAccessToken . token _type + " " + this . oauthAccessToken . access _token ,
} ;
} catch ( e ) {
throw new Error ( "The oauth config is invalid. " + e . message ) ;
}
}
2021-10-30 19:37:15 +02:00
const httpsAgentOptions = {
maxCachedSessions : 0 , // Use Custom agent to disable session reuse (https://github.com/nodejs/node/issues/3940)
rejectUnauthorized : ! this . getIgnoreTls ( ) ,
2023-11-17 09:21:08 +02:00
secureOptions : crypto . constants . SSL _OP _LEGACY _SERVER _CONNECT ,
2021-10-30 19:37:15 +02:00
} ;
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] Prepare Options for axios ` ) ;
2021-11-07 18:15:36 +02:00
2022-09-26 23:09:10 +02:00
let contentType = null ;
2022-09-26 21:52:43 +02:00
let bodyValue = null ;
2023-01-14 15:33:21 +02:00
2023-02-25 11:28:32 +02:00
if ( this . body && ( typeof this . body === "string" && this . body . trim ( ) . length > 0 ) ) {
if ( ! this . httpBodyEncoding || this . httpBodyEncoding === "json" ) {
try {
bodyValue = JSON . parse ( this . body ) ;
contentType = "application/json" ;
} catch ( e ) {
throw new Error ( "Your JSON body is invalid. " + e . message ) ;
}
2023-09-09 12:05:25 +02:00
} else if ( this . httpBodyEncoding === "form" ) {
bodyValue = this . body ;
contentType = "application/x-www-form-urlencoded" ;
2023-02-25 11:28:32 +02:00
} else if ( this . httpBodyEncoding === "xml" ) {
bodyValue = this . body ;
contentType = "text/xml; charset=utf-8" ;
}
2022-08-12 02:57:03 +02:00
}
2022-12-04 16:55:05 +02:00
// Axios Options
2021-10-02 16:48:27 +02:00
const options = {
url : this . url ,
method : ( this . method || "get" ) . toLowerCase ( ) ,
2023-08-08 12:29:48 +02:00
timeout : this . timeout * 1000 ,
2021-07-30 13:18:26 +02:00
headers : {
2021-12-16 09:09:10 +02:00
"Accept" : "text/html,application/xhtml+xml,application/xml;q=0.9,image/avif,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9" ,
2023-01-20 11:29:56 +02:00
... ( contentType ? { "Content-Type" : contentType } : { } ) ,
2021-11-04 11:12:06 +02:00
... ( basicAuthHeader ) ,
2023-08-02 09:40:19 +02:00
... ( oauth2AuthHeader ) ,
2023-01-20 11:29:56 +02:00
... ( this . headers ? JSON . parse ( this . headers ) : { } )
2021-07-30 13:18:26 +02:00
} ,
2021-08-05 13:04:38 +02:00
maxRedirects : this . maxredirects ,
validateStatus : ( status ) => {
return checkStatusCode ( status , this . getAcceptedStatuscodes ( ) ) ;
} ,
2023-11-22 10:03:03 +02:00
signal : axiosAbortSignal ( ( this . timeout + 10 ) * 1000 ) ,
2021-10-02 16:48:27 +02:00
} ;
2021-11-07 15:00:47 +02:00
2023-02-25 11:28:32 +02:00
if ( bodyValue ) {
options . data = bodyValue ;
}
2024-08-24 22:57:31 +02:00
if ( this . cacheBust ) {
const randomFloatString = Math . random ( ) . toString ( 36 ) ;
const cacheBust = randomFloatString . substring ( 2 ) ;
options . params = {
uptime _kuma _cachebuster : cacheBust ,
} ;
}
2021-10-30 19:37:15 +02:00
if ( this . proxy _id ) {
const proxy = await R . load ( "proxy" , this . proxy _id ) ;
if ( proxy && proxy . active ) {
2021-11-04 13:32:16 +02:00
const { httpAgent , httpsAgent } = Proxy . createAgents ( proxy , {
httpsAgentOptions : httpsAgentOptions ,
} ) ;
2021-10-30 19:37:15 +02:00
options . proxy = false ;
2021-11-04 13:32:16 +02:00
options . httpAgent = httpAgent ;
options . httpsAgent = httpsAgent ;
2021-10-30 19:37:15 +02:00
}
}
if ( ! options . httpsAgent ) {
2023-11-29 11:25:33 +02:00
let jar = new CookieJar ( ) ;
let httpsCookieAgentOptions = {
... httpsAgentOptions ,
cookies : { jar }
} ;
options . httpsAgent = new HttpsCookieAgent ( httpsCookieAgentOptions ) ;
2021-10-30 19:37:15 +02:00
}
2023-01-04 09:37:30 +02:00
if ( this . auth _method === "mtls" ) {
if ( this . tlsCert !== null && this . tlsCert !== "" ) {
options . httpsAgent . options . cert = Buffer . from ( this . tlsCert ) ;
}
if ( this . tlsCa !== null && this . tlsCa !== "" ) {
options . httpsAgent . options . ca = Buffer . from ( this . tlsCa ) ;
}
if ( this . tlsKey !== null && this . tlsKey !== "" ) {
options . httpsAgent . options . key = Buffer . from ( this . tlsKey ) ;
}
}
2024-04-24 08:37:17 +02:00
let tlsInfo = { } ;
// Store tlsInfo when secureConnect event is emitted
// The keylog event listener is a workaround to access the tlsSocket
options . httpsAgent . once ( "keylog" , async ( line , tlsSocket ) => {
tlsSocket . once ( "secureConnect" , async ( ) => {
tlsInfo = checkCertificate ( tlsSocket ) ;
tlsInfo . valid = tlsSocket . authorized || false ;
await this . handleTlsInfo ( tlsInfo ) ;
} ) ;
2024-04-06 12:43:08 +02:00
} ) ;
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] Axios Options: ${ JSON . stringify ( options ) } ` ) ;
log . debug ( "monitor" , ` [ ${ this . name } ] Axios Request ` ) ;
2021-10-30 19:37:15 +02:00
2022-12-26 15:00:46 +02:00
// Make Request
let res = await this . makeAxiosRequest ( options ) ;
2022-05-13 19:58:23 +02:00
2021-09-17 08:42:19 +02:00
bean . msg = ` ${ res . status } - ${ res . statusText } ` ;
2021-06-27 10:10:55 +02:00
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
2021-07-01 11:19:28 +02:00
2024-04-24 08:37:17 +02:00
// fallback for if kelog event is not emitted, but we may still have tlsInfo,
// e.g. if the connection is made through a proxy
if ( this . getUrl ( ) ? . protocol === "https:" && tlsInfo . valid === undefined ) {
const tlsSocket = res . request . res . socket ;
2021-10-27 09:33:15 +02:00
2024-04-24 08:37:17 +02:00
if ( tlsSocket ) {
tlsInfo = checkCertificate ( tlsSocket ) ;
tlsInfo . valid = tlsSocket . authorized || false ;
2021-10-27 09:33:15 +02:00
2024-04-24 08:37:17 +02:00
await this . handleTlsInfo ( tlsInfo ) ;
2021-07-23 06:58:05 +02:00
}
2021-07-21 06:09:09 +02:00
}
2021-07-01 11:19:28 +02:00
2022-04-17 09:43:03 +02:00
if ( process . env . UPTIME _KUMA _LOG _RESPONSE _BODY _MONITOR _ID === this . id ) {
2022-04-13 17:33:37 +02:00
log . info ( "monitor" , res . data ) ;
2021-10-15 12:36:40 +02:00
}
2021-07-23 06:58:05 +02:00
2021-07-01 11:19:28 +02:00
if ( this . type === "http" ) {
2021-07-24 05:42:14 +02:00
bean . status = UP ;
2023-07-13 17:37:26 +02:00
} else if ( this . type === "keyword" ) {
2021-07-01 11:19:28 +02:00
2021-07-12 04:52:41 +02:00
let data = res . data ;
// Convert to string for object/array
if ( typeof data !== "string" ) {
2021-09-17 08:42:19 +02:00
data = JSON . stringify ( data ) ;
2021-07-12 04:52:41 +02:00
}
2023-04-06 22:30:38 +02:00
let keywordFound = data . includes ( this . keyword ) ;
if ( keywordFound === ! this . isInvertKeyword ( ) ) {
bean . msg += ", keyword " + ( keywordFound ? "is" : "not" ) + " found" ;
2021-07-24 05:42:14 +02:00
bean . status = UP ;
2021-07-01 11:19:28 +02:00
} else {
2023-04-08 18:23:29 +02:00
data = data . replace ( /<[^>]*>?|[\n\r]|\s+/gm , " " ) . trim ( ) ;
2022-04-01 09:26:50 +02:00
if ( data . length > 50 ) {
2022-01-24 07:31:49 +02:00
data = data . substring ( 0 , 47 ) + "..." ;
}
2023-04-06 02:10:21 +02:00
throw new Error ( bean . msg + ", but keyword is " +
2023-04-06 22:30:38 +02:00
( keywordFound ? "present" : "not" ) + " in [" + data + "]" ) ;
2021-07-01 11:19:28 +02:00
}
2023-07-13 17:37:26 +02:00
} else if ( this . type === "json-query" ) {
let data = res . data ;
2024-06-07 02:52:33 +02:00
const { status , response } = await evaluateJsonQuery ( data , this . jsonPath , this . jsonPathOperator , this . expectedValue ) ;
2021-07-01 11:19:28 +02:00
2024-06-12 22:25:25 +02:00
if ( status ) {
2024-06-12 23:24:01 +02:00
bean . status = UP ;
bean . msg = ` JSON query passes (comparing ${ response } ${ this . jsonPathOperator } ${ this . expectedValue } ) ` ;
2024-06-12 22:25:25 +02:00
} else {
throw new Error ( ` JSON query does not pass (comparing ${ response } ${ this . jsonPathOperator } ${ this . expectedValue } ) ` ) ;
}
2023-07-13 17:37:26 +02:00
2021-07-01 11:19:28 +02:00
}
2021-07-01 08:03:06 +02:00
} else if ( this . type === "port" ) {
bean . ping = await tcping ( this . hostname , this . port ) ;
2021-09-17 08:42:19 +02:00
bean . msg = "" ;
2021-07-24 05:42:14 +02:00
bean . status = UP ;
2021-07-01 11:00:23 +02:00
} else if ( this . type === "ping" ) {
2022-07-14 09:32:51 +02:00
bean . ping = await ping ( this . hostname , this . packetSize ) ;
2021-09-17 08:42:19 +02:00
bean . msg = "" ;
2021-07-24 05:42:14 +02:00
bean . status = UP ;
2021-09-30 18:09:43 +02:00
} else if ( this . type === "push" ) { // Type: Push
2022-05-27 06:45:56 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] Checking monitor at ${ dayjs ( ) . format ( "YYYY-MM-DD HH:mm:ss.SSS" ) } ` ) ;
2022-03-28 23:28:50 +02:00
const bufferTime = 1000 ; // 1s buffer to accommodate clock differences
2022-05-28 17:19:58 +02:00
2022-03-28 23:28:50 +02:00
if ( previousBeat ) {
const msSinceLastBeat = dayjs . utc ( ) . valueOf ( ) - dayjs . utc ( previousBeat . time ) . valueOf ( ) ;
2022-05-28 17:22:44 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] msSinceLastBeat = ${ msSinceLastBeat } ` ) ;
2022-05-29 04:57:45 +02:00
// If the previous beat was down or pending we use the regular
// beatInterval/retryInterval in the setTimeout further below
2022-06-14 07:05:58 +02:00
if ( previousBeat . status !== ( this . isUpsideDown ( ) ? DOWN : UP ) || msSinceLastBeat > beatInterval * 1000 + bufferTime ) {
2023-11-24 12:11:36 +02:00
bean . duration = Math . round ( msSinceLastBeat / 1000 ) ;
2022-03-28 23:28:50 +02:00
throw new Error ( "No heartbeat in the time window" ) ;
} else {
let timeout = beatInterval * 1000 - msSinceLastBeat ;
if ( timeout < 0 ) {
timeout = bufferTime ;
} else {
timeout += bufferTime ;
}
// No need to insert successful heartbeat for push type, so end here
retries = 0 ;
2022-05-27 06:45:56 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] timeout = ${ timeout } ` ) ;
2023-05-19 12:52:00 +02:00
this . heartbeatInterval = setTimeout ( safeBeat , timeout ) ;
2022-03-28 23:28:50 +02:00
return ;
}
2022-05-29 04:57:45 +02:00
} else {
2023-11-24 12:11:36 +02:00
bean . duration = beatInterval ;
2022-05-29 04:57:45 +02:00
throw new Error ( "No heartbeat in the time window" ) ;
2021-09-30 18:09:43 +02:00
}
2021-09-27 11:17:57 +02:00
} else if ( this . type === "steam" ) {
const steamApiUrl = "https://api.steampowered.com/IGameServersService/GetServerList/v1/" ;
2024-10-09 01:43:44 +02:00
const steamAPIKey = await setting ( "steamAPIKey" ) ;
2021-09-27 11:17:57 +02:00
const filter = ` addr \\ ${ this . hostname } : ${ this . port } ` ;
2021-10-18 11:11:41 +02:00
if ( ! steamAPIKey ) {
throw new Error ( "Steam API Key not found" ) ;
}
2022-06-23 09:54:33 +02:00
let res = await axios . get ( steamApiUrl , {
2023-08-08 12:29:48 +02:00
timeout : this . timeout * 1000 ,
2021-09-27 11:17:57 +02:00
headers : {
"Accept" : "*/*" ,
} ,
2023-12-10 20:01:56 +02:00
httpsAgent : new https . Agent ( {
2021-09-27 11:17:57 +02:00
maxCachedSessions : 0 , // Use Custom agent to disable session reuse (https://github.com/nodejs/node/issues/3940)
2021-11-04 03:46:43 +02:00
rejectUnauthorized : ! this . getIgnoreTls ( ) ,
2023-11-17 09:21:08 +02:00
secureOptions : crypto . constants . SSL _OP _LEGACY _SERVER _CONNECT ,
2021-09-27 11:17:57 +02:00
} ) ,
2023-12-10 20:01:56 +02:00
httpAgent : new http . Agent ( {
2022-07-18 16:33:35 +02:00
maxCachedSessions : 0 ,
} ) ,
2021-09-27 11:17:57 +02:00
maxRedirects : this . maxredirects ,
validateStatus : ( status ) => {
return checkStatusCode ( status , this . getAcceptedStatuscodes ( ) ) ;
} ,
params : {
filter : filter ,
2021-10-18 11:11:41 +02:00
key : steamAPIKey ,
2021-09-27 11:17:57 +02:00
}
} ) ;
2021-10-18 11:02:05 +02:00
if ( res . data . response && res . data . response . servers && res . data . response . servers . length > 0 ) {
bean . status = UP ;
bean . msg = res . data . response . servers [ 0 ] . name ;
2021-09-27 11:17:57 +02:00
2021-10-18 11:02:05 +02:00
try {
2022-07-14 09:32:51 +02:00
bean . ping = await ping ( this . hostname , this . packetSize ) ;
2021-10-18 11:02:05 +02:00
} catch ( _ ) { }
2021-09-27 11:17:57 +02:00
} else {
2021-10-18 11:02:05 +02:00
throw new Error ( "Server not found on Steam" ) ;
2021-09-27 11:17:57 +02:00
}
2023-01-08 10:22:36 +02:00
} else if ( this . type === "gamedig" ) {
try {
const state = await Gamedig . query ( {
type : this . game ,
host : this . hostname ,
2023-01-24 18:19:54 +02:00
port : this . port ,
2023-08-07 21:14:21 +02:00
givenPortOnly : this . getGameDigGivenPortOnly ( ) ,
2023-01-08 10:22:36 +02:00
} ) ;
bean . msg = state . name ;
bean . status = UP ;
bean . ping = state . ping ;
} catch ( e ) {
2023-01-24 17:03:01 +02:00
throw new Error ( e . message ) ;
2023-01-08 10:22:36 +02:00
}
2022-01-13 18:17:07 +02:00
} else if ( this . type === "docker" ) {
2023-01-24 17:40:24 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] Prepare Options for Axios ` ) ;
2022-01-13 18:17:07 +02:00
const options = {
url : ` /containers/ ${ this . docker _container } /json ` ,
2023-01-18 03:53:04 +02:00
timeout : this . interval * 1000 * 0.8 ,
2022-01-13 18:17:07 +02:00
headers : {
"Accept" : "*/*" ,
} ,
2023-12-10 20:01:56 +02:00
httpsAgent : new https . Agent ( {
2022-01-13 18:17:07 +02:00
maxCachedSessions : 0 , // Use Custom agent to disable session reuse (https://github.com/nodejs/node/issues/3940)
2023-01-18 03:53:04 +02:00
rejectUnauthorized : ! this . getIgnoreTls ( ) ,
2023-11-17 09:21:08 +02:00
secureOptions : crypto . constants . SSL _OP _LEGACY _SERVER _CONNECT ,
2023-01-18 03:53:04 +02:00
} ) ,
2023-12-10 20:01:56 +02:00
httpAgent : new http . Agent ( {
2023-01-18 03:53:04 +02:00
maxCachedSessions : 0 ,
2022-01-13 18:17:07 +02:00
} ) ,
} ;
2023-11-12 14:32:40 +02:00
const dockerHost = await R . load ( "docker_host" , this . docker _host ) ;
if ( ! dockerHost ) {
throw new Error ( "Failed to load docker host config" ) ;
}
2022-07-22 17:57:40 +02:00
if ( dockerHost . _dockerType === "socket" ) {
options . socketPath = dockerHost . _dockerDaemon ;
} else if ( dockerHost . _dockerType === "tcp" ) {
2022-10-04 10:19:56 +02:00
options . baseURL = DockerHost . patchDockerURL ( dockerHost . _dockerDaemon ) ;
2023-12-10 20:01:56 +02:00
options . httpsAgent = new https . Agent (
2023-08-04 16:49:33 +02:00
DockerHost . getHttpsAgentOptions ( dockerHost . _dockerType , options . baseURL )
) ;
2022-01-22 03:57:37 +02:00
}
2023-01-24 17:40:24 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] Axios Request ` ) ;
2022-01-13 18:17:07 +02:00
let res = await axios . request ( options ) ;
2023-05-17 18:52:28 +02:00
2022-01-13 18:17:07 +02:00
if ( res . data . State . Running ) {
2023-05-18 09:55:33 +02:00
if ( res . data . State . Health && res . data . State . Health . Status !== "healthy" ) {
bean . status = PENDING ;
bean . msg = res . data . State . Health . Status ;
} else {
bean . status = UP ;
bean . msg = res . data . State . Health ? res . data . State . Health . Status : res . data . State . Status ;
}
2023-01-03 16:07:14 +02:00
} else {
throw Error ( "Container State is " + res . data . State . Status ) ;
2022-01-13 18:17:07 +02:00
}
2022-05-12 19:48:03 +02:00
} else if ( this . type === "sqlserver" ) {
let startTime = dayjs ( ) . valueOf ( ) ;
2023-10-15 18:38:56 +02:00
await mssqlQuery ( this . databaseConnectionString , this . databaseQuery || "SELECT 1" ) ;
2022-05-12 19:48:03 +02:00
bean . msg = "" ;
bean . status = UP ;
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
2022-08-03 07:00:39 +02:00
} else if ( this . type === "grpc-keyword" ) {
let startTime = dayjs ( ) . valueOf ( ) ;
const options = {
grpcUrl : this . grpcUrl ,
grpcProtobufData : this . grpcProtobuf ,
grpcServiceName : this . grpcServiceName ,
grpcEnableTls : this . grpcEnableTls ,
grpcMethod : this . grpcMethod ,
grpcBody : this . grpcBody ,
} ;
const response = await grpcQuery ( options ) ;
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
log . debug ( "monitor:" , ` gRPC response: ${ JSON . stringify ( response ) } ` ) ;
let responseData = response . data ;
if ( responseData . length > 50 ) {
2022-12-28 17:31:33 +02:00
responseData = responseData . toString ( ) . substring ( 0 , 47 ) + "..." ;
2022-08-03 07:00:39 +02:00
}
if ( response . code !== 1 ) {
bean . status = DOWN ;
bean . msg = ` Error in send gRPC ${ response . code } ${ response . errorMessage } ` ;
} else {
2023-04-06 22:30:38 +02:00
let keywordFound = response . data . toString ( ) . includes ( this . keyword ) ;
if ( keywordFound === ! this . isInvertKeyword ( ) ) {
2022-08-03 07:00:39 +02:00
bean . status = UP ;
2023-04-06 22:30:38 +02:00
bean . msg = ` ${ responseData } , keyword [ ${ this . keyword } ] ${ keywordFound ? "is" : "not" } found ` ;
2022-08-03 07:00:39 +02:00
} else {
2023-04-06 22:30:38 +02:00
log . debug ( "monitor:" , ` GRPC response [ ${ response . data } ] + ", but keyword [ ${ this . keyword } ] is ${ keywordFound ? "present" : "not" } in [" + ${ response . data } + "]" ` ) ;
2022-08-03 07:00:39 +02:00
bean . status = DOWN ;
2023-04-06 22:30:38 +02:00
bean . msg = ` , but keyword [ ${ this . keyword } ] is ${ keywordFound ? "present" : "not" } in [" + ${ responseData } + "] ` ;
2022-08-03 07:00:39 +02:00
}
}
2022-06-15 19:12:47 +02:00
} else if ( this . type === "postgres" ) {
let startTime = dayjs ( ) . valueOf ( ) ;
2023-10-15 18:38:56 +02:00
await postgresQuery ( this . databaseConnectionString , this . databaseQuery || "SELECT 1" ) ;
2022-06-15 19:12:47 +02:00
2022-10-02 02:52:53 +02:00
bean . msg = "" ;
bean . status = UP ;
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
} else if ( this . type === "mysql" ) {
let startTime = dayjs ( ) . valueOf ( ) ;
2023-10-15 18:38:56 +02:00
// Use `radius_password` as `password` field, since there are too many unnecessary fields
// TODO: rename `radius_password` to `password` later for general use
let mysqlPassword = this . radiusPassword ;
bean . msg = await mysqlQuery ( this . databaseConnectionString , this . databaseQuery || "SELECT 1" , mysqlPassword ) ;
2022-05-12 19:48:03 +02:00
bean . status = UP ;
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
2022-05-12 11:48:38 +02:00
} else if ( this . type === "radius" ) {
let startTime = dayjs ( ) . valueOf ( ) ;
2022-10-12 18:32:05 +02:00
// Handle monitors that were created before the
// update and as such don't have a value for
// this.port.
let port ;
if ( this . port == null ) {
port = 1812 ;
} else {
port = this . port ;
}
2023-07-27 11:42:22 +02:00
const resp = await radius (
this . hostname ,
this . radiusUsername ,
this . radiusPassword ,
this . radiusCalledStationId ,
this . radiusCallingStationId ,
this . radiusSecret ,
port ,
this . interval * 1000 * 0.4 ,
) ;
bean . msg = resp . code ;
bean . status = UP ;
2022-05-12 11:48:38 +02:00
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
2023-01-05 16:58:24 +02:00
} else if ( this . type === "redis" ) {
let startTime = dayjs ( ) . valueOf ( ) ;
2024-05-19 23:34:01 +02:00
bean . msg = await redisPingAsync ( this . databaseConnectionString , ! this . ignoreTls ) ;
2023-01-05 16:58:24 +02:00
bean . status = UP ;
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
2023-01-27 12:25:57 +02:00
} else if ( this . type in UptimeKumaServer . monitorTypeList ) {
let startTime = dayjs ( ) . valueOf ( ) ;
const monitorType = UptimeKumaServer . monitorTypeList [ this . type ] ;
2023-06-27 09:54:33 +02:00
await monitorType . check ( this , bean , UptimeKumaServer . getInstance ( ) ) ;
2023-01-27 12:25:57 +02:00
if ( ! bean . ping ) {
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
}
2023-07-17 10:15:44 +02:00
} else if ( this . type === "kafka-producer" ) {
let startTime = dayjs ( ) . valueOf ( ) ;
bean . msg = await kafkaProducerAsync (
JSON . parse ( this . kafkaProducerBrokers ) ,
this . kafkaProducerTopic ,
this . kafkaProducerMessage ,
{
allowAutoTopicCreation : this . kafkaProducerAllowAutoTopicCreation ,
ssl : this . kafkaProducerSsl ,
clientId : ` Uptime-Kuma/ ${ version } ` ,
interval : this . interval ,
} ,
JSON . parse ( this . kafkaProducerSaslOptions ) ,
) ;
bean . status = UP ;
bean . ping = dayjs ( ) . valueOf ( ) - startTime ;
2021-09-30 18:09:43 +02:00
} else {
2023-01-27 12:25:57 +02:00
throw new Error ( "Unknown Monitor Type" ) ;
2021-06-27 10:10:55 +02:00
}
2021-07-30 18:01:04 +02:00
if ( this . isUpsideDown ( ) ) {
bean . status = flipStatus ( bean . status ) ;
if ( bean . status === DOWN ) {
throw new Error ( "Flip UP to DOWN" ) ;
}
}
2021-07-19 18:23:06 +02:00
retries = 0 ;
2021-06-27 10:10:55 +02:00
} catch ( error ) {
2024-06-23 17:46:52 +02:00
2023-11-21 17:56:17 +02:00
if ( error ? . name === "CanceledError" ) {
bean . msg = ` timeout by AbortSignal ( ${ this . timeout } s) ` ;
} else {
bean . msg = error . message ;
}
2021-07-30 18:01:04 +02:00
// If UP come in here, it must be upside down mode
// Just reset the retries
if ( this . isUpsideDown ( ) && bean . status === UP ) {
retries = 0 ;
} else if ( ( this . maxretries > 0 ) && ( retries < this . maxretries ) ) {
2021-07-19 18:23:06 +02:00
retries ++ ;
2021-07-24 05:42:14 +02:00
bean . status = PENDING ;
2023-11-24 12:11:36 +02:00
} else {
// Continue counting retries during DOWN
retries ++ ;
2021-07-19 18:23:06 +02:00
}
2021-06-27 10:10:55 +02:00
}
2023-11-24 12:11:36 +02:00
bean . retries = retries ;
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] Check isImportant ` ) ;
2021-10-16 11:28:03 +02:00
let isImportant = Monitor . isImportantBeat ( isFirstBeat , previousBeat ? . status , bean . status ) ;
2021-07-24 05:42:14 +02:00
2021-07-20 11:50:33 +02:00
// Mark as important if status changed, ignore pending pings,
// Don't notify if disrupted changes to up
2021-07-24 05:42:14 +02:00
if ( isImportant ) {
2021-06-29 10:06:20 +02:00
bean . important = true ;
2021-11-07 15:00:47 +02:00
2022-01-23 16:22:00 +02:00
if ( Monitor . isImportantForNotification ( isFirstBeat , previousBeat ? . status , bean . status ) ) {
2022-04-30 13:40:34 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] sendNotification ` ) ;
2022-01-23 16:22:00 +02:00
await Monitor . sendNotification ( isFirstBeat , this , bean ) ;
2022-04-30 14:57:08 +02:00
} else {
2022-04-30 13:40:34 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] will not sendNotification because it is (or was) under maintenance ` ) ;
2022-01-23 16:22:00 +02:00
}
2021-10-27 05:39:46 +02:00
2022-06-15 16:56:26 +02:00
// Reset down count
bean . downCount = 0 ;
2022-01-23 16:22:57 +02:00
2021-10-27 05:39:46 +02:00
// Clear Status Page Cache
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] apicache clear ` ) ;
2021-10-27 05:39:46 +02:00
apicache . clear ( ) ;
2024-03-15 16:02:55 +02:00
await UptimeKumaServer . getInstance ( ) . sendMaintenanceListByUserID ( this . user _id ) ;
2022-10-15 14:15:50 +02:00
2021-06-29 10:06:20 +02:00
} else {
bean . important = false ;
2022-01-23 16:22:57 +02:00
if ( bean . status === DOWN && this . resendInterval > 0 ) {
2022-06-15 16:56:26 +02:00
++ bean . downCount ;
if ( bean . downCount >= this . resendInterval ) {
2022-01-23 16:22:57 +02:00
// Send notification again, because we are still DOWN
2022-06-15 16:56:26 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] sendNotification again: Down Count: ${ bean . downCount } | Resend Interval: ${ this . resendInterval } ` ) ;
2022-01-23 16:22:57 +02:00
await Monitor . sendNotification ( isFirstBeat , this , bean ) ;
2022-06-15 16:56:26 +02:00
// Reset down count
bean . downCount = 0 ;
2022-01-23 16:22:57 +02:00
}
}
2021-06-29 10:06:20 +02:00
}
2021-07-24 05:42:14 +02:00
if ( bean . status === UP ) {
2022-06-13 15:15:47 +02:00
log . debug ( "monitor" , ` Monitor # ${ this . id } ' ${ this . name } ': Successful Response: ${ bean . ping } ms | Interval: ${ beatInterval } seconds | Type: ${ this . type } ` ) ;
2021-07-24 05:42:14 +02:00
} else if ( bean . status === PENDING ) {
2021-09-29 11:20:35 +02:00
if ( this . retryInterval > 0 ) {
2021-09-11 18:54:55 +02:00
beatInterval = this . retryInterval ;
}
2022-04-13 17:33:37 +02:00
log . warn ( "monitor" , ` Monitor # ${ this . id } ' ${ this . name } ': Pending: ${ bean . msg } | Max retries: ${ this . maxretries } | Retry: ${ retries } | Retry Interval: ${ beatInterval } seconds | Type: ${ this . type } ` ) ;
2022-01-23 16:22:00 +02:00
} else if ( bean . status === MAINTENANCE ) {
2022-04-30 13:40:34 +02:00
log . warn ( "monitor" , ` Monitor # ${ this . id } ' ${ this . name } ': Under Maintenance | Type: ${ this . type } ` ) ;
2021-07-21 00:41:38 +02:00
} else {
2022-06-15 16:56:26 +02:00
log . warn ( "monitor" , ` Monitor # ${ this . id } ' ${ this . name } ': Failing: ${ bean . msg } | Interval: ${ beatInterval } seconds | Type: ${ this . type } | Down Count: ${ bean . downCount } | Resend Interval: ${ this . resendInterval } ` ) ;
2021-07-21 00:41:38 +02:00
}
2023-08-31 23:19:21 +02:00
// Calculate uptime
let uptimeCalculator = await UptimeCalculator . getUptimeCalculator ( this . id ) ;
let endTimeDayjs = await uptimeCalculator . update ( bean . status , parseFloat ( bean . ping ) ) ;
bean . end _time = R . isoDateTimeMillis ( endTimeDayjs ) ;
// Send to frontend
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] Send to socket ` ) ;
2021-06-29 10:06:20 +02:00
io . to ( this . user _id ) . emit ( "heartbeat" , bean . toJSON ( ) ) ;
2021-09-17 08:42:19 +02:00
Monitor . sendStats ( io , this . id , this . user _id ) ;
2021-06-29 10:06:20 +02:00
2023-08-31 23:19:21 +02:00
// Store to database
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] Store ` ) ;
2021-08-23 12:52:24 +02:00
await R . store ( bean ) ;
2021-11-07 15:00:47 +02:00
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] prometheus.update ` ) ;
2024-10-09 01:17:11 +02:00
this . prometheus ? . update ( bean , tlsInfo ) ;
2021-08-23 12:52:24 +02:00
2021-06-29 10:06:20 +02:00
previousBeat = bean ;
2021-08-23 12:52:24 +02:00
2021-09-08 14:00:16 +02:00
if ( ! this . isStop ) {
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , ` [ ${ this . name } ] SetTimeout for next check. ` ) ;
2023-09-21 14:11:04 +02:00
let intervalRemainingMs = Math . max (
1 ,
beatInterval * 1000 - dayjs ( ) . diff ( dayjs . utc ( bean . time ) )
) ;
log . debug ( "monitor" , ` [ ${ this . name } ] Next heartbeat in: ${ intervalRemainingMs } ms ` ) ;
this . heartbeatInterval = setTimeout ( safeBeat , intervalRemainingMs ) ;
2021-11-07 15:00:47 +02:00
} else {
2022-04-13 17:33:37 +02:00
log . info ( "monitor" , ` [ ${ this . name } ] isStop = true, no next check. ` ) ;
2021-09-08 14:00:16 +02:00
}
2021-09-17 08:42:19 +02:00
} ;
2021-06-25 15:55:49 +02:00
2023-08-11 09:46:41 +02:00
/ * *
* Get a heartbeat and handle errors7
* @ returns { void }
* /
2021-10-27 08:08:44 +02:00
const safeBeat = async ( ) => {
try {
await beat ( ) ;
} catch ( e ) {
console . trace ( e ) ;
2022-05-06 08:41:34 +02:00
UptimeKumaServer . errorLog ( e , false ) ;
2022-04-13 17:33:37 +02:00
log . error ( "monitor" , "Please report to https://github.com/louislam/uptime-kuma/issues" ) ;
2021-10-27 08:08:44 +02:00
if ( ! this . isStop ) {
2022-04-13 17:33:37 +02:00
log . info ( "monitor" , "Try to restart the monitor" ) ;
2021-10-27 08:08:44 +02:00
this . heartbeatInterval = setTimeout ( safeBeat , this . interval * 1000 ) ;
}
}
} ;
2021-09-30 18:09:43 +02:00
// Delay Push Type
if ( this . type === "push" ) {
setTimeout ( ( ) => {
2021-10-27 08:08:44 +02:00
safeBeat ( ) ;
2021-09-30 18:09:43 +02:00
} , this . interval * 1000 ) ;
} else {
2021-10-27 08:08:44 +02:00
safeBeat ( ) ;
2021-09-30 18:09:43 +02:00
}
2021-06-25 15:55:49 +02:00
}
2023-01-06 00:19:05 +02:00
/ * *
* Make a request using axios
2023-08-11 09:46:41 +02:00
* @ param { object } options Options for Axios
2023-01-06 00:19:05 +02:00
* @ param { boolean } finalCall Should this be the final call i . e
2023-08-11 09:46:41 +02:00
* don ' t retry on failure
* @ returns { object } Axios response
2023-01-06 00:19:05 +02:00
* /
2022-12-26 15:00:46 +02:00
async makeAxiosRequest ( options , finalCall = false ) {
try {
let res ;
if ( this . auth _method === "ntlm" ) {
options . httpsAgent . keepAlive = true ;
res = await httpNtlm ( options , {
username : this . basic _auth _user ,
password : this . basic _auth _pass ,
domain : this . authDomain ,
workstation : this . authWorkstation ? this . authWorkstation : undefined
} ) ;
} else {
res = await axios . request ( options ) ;
}
return res ;
2023-12-17 11:21:07 +02:00
} catch ( error ) {
/ * *
* Make a single attempt to obtain an new access token in the event that
* the recent api request failed for authentication purposes
* /
if ( this . auth _method === "oauth2-cc" && error . response . status === 401 && ! finalCall ) {
this . oauthAccessToken = await this . makeOidcTokenClientCredentialsRequest ( ) ;
let oauth2AuthHeader = {
"Authorization" : this . oauthAccessToken . token _type + " " + this . oauthAccessToken . access _token ,
} ;
options . headers = { ... ( options . headers ) ,
... ( oauth2AuthHeader )
} ;
return this . makeAxiosRequest ( options , true ) ;
}
2022-12-26 15:00:46 +02:00
// Fix #2253
// Read more: https://stackoverflow.com/questions/1759956/curl-error-18-transfer-closed-with-outstanding-read-data-remaining
2023-12-17 11:21:07 +02:00
if ( ! finalCall && typeof error . message === "string" && error . message . includes ( "maxContentLength size of -1 exceeded" ) ) {
2022-12-26 15:00:46 +02:00
log . debug ( "monitor" , "makeAxiosRequest with gzip" ) ;
options . headers [ "Accept-Encoding" ] = "gzip, deflate" ;
return this . makeAxiosRequest ( options , true ) ;
} else {
2023-12-17 11:21:07 +02:00
if ( typeof error . message === "string" && error . message . includes ( "maxContentLength size of -1 exceeded" ) ) {
error . message = "response timeout: incomplete response within a interval" ;
2022-12-26 15:00:46 +02:00
}
2023-12-17 11:21:07 +02:00
throw error ;
2022-12-26 15:00:46 +02:00
}
}
}
2023-08-11 09:46:41 +02:00
/ * *
* Stop monitor
2024-01-30 18:53:21 +02:00
* @ returns { Promise < void > }
2023-08-11 09:46:41 +02:00
* /
2023-02-23 18:16:49 +02:00
async stop ( ) {
2021-08-23 12:52:24 +02:00
clearTimeout ( this . heartbeatInterval ) ;
2021-09-08 14:00:16 +02:00
this . isStop = true ;
2022-01-07 06:26:26 +02:00
2023-02-24 11:12:57 +02:00
this . prometheus ? . remove ( ) ;
2021-06-25 15:55:49 +02:00
}
2021-06-30 15:04:58 +02:00
2022-04-21 19:30:04 +02:00
/ * *
2023-01-24 11:14:16 +02:00
* Get prometheus instance
2023-08-11 09:46:41 +02:00
* @ returns { Prometheus | undefined } Current prometheus instance
2022-04-21 19:30:04 +02:00
* /
2023-01-24 11:14:16 +02:00
getPrometheus ( ) {
return this . prometheus ;
2021-06-25 15:55:49 +02:00
}
2021-06-30 15:04:58 +02:00
2021-07-30 05:23:04 +02:00
/ * *
* Helper Method :
* returns URL object for further usage
* returns null if url is invalid
2023-08-11 09:46:41 +02:00
* @ returns { ( null | URL ) } Monitor URL
2021-07-30 05:23:04 +02:00
* /
2021-07-22 10:04:32 +02:00
getUrl ( ) {
try {
return new URL ( this . url ) ;
} catch ( _ ) {
return null ;
}
}
2023-09-16 20:40:08 +02:00
/ * *
* Example : http : or https :
2023-10-02 23:39:17 +02:00
* @ returns { ( null | string ) } URL ' s protocol
2023-09-16 20:40:08 +02:00
* /
getURLProtocol ( ) {
const url = this . getUrl ( ) ;
if ( url ) {
return this . getUrl ( ) . protocol ;
} else {
return null ;
}
}
2021-07-30 05:23:04 +02:00
/ * *
* Store TLS info to database
2023-08-11 09:46:41 +02:00
* @ param { object } checkCertificateResult Certificate to update
* @ returns { Promise < object > } Updated certificate
2021-07-30 05:23:04 +02:00
* /
2021-07-22 10:04:32 +02:00
async updateTlsInfo ( checkCertificateResult ) {
2022-04-13 18:30:32 +02:00
let tlsInfoBean = await R . findOne ( "monitor_tls_info" , "monitor_id = ?" , [
2021-07-30 13:18:26 +02:00
this . id ,
2021-07-22 10:04:32 +02:00
] ) ;
2021-10-27 10:12:18 +02:00
2022-04-13 18:30:32 +02:00
if ( tlsInfoBean == null ) {
tlsInfoBean = R . dispense ( "monitor_tls_info" ) ;
tlsInfoBean . monitor _id = this . id ;
2021-10-27 10:12:18 +02:00
} else {
2021-10-27 10:03:16 +02:00
2021-10-27 10:12:18 +02:00
// Clear sent history if the cert changed.
try {
2022-04-13 18:30:32 +02:00
let oldCertInfo = JSON . parse ( tlsInfoBean . info _json ) ;
2021-10-27 10:03:16 +02:00
2021-10-27 10:12:18 +02:00
let isValidObjects = oldCertInfo && oldCertInfo . certInfo && checkCertificateResult && checkCertificateResult . certInfo ;
if ( isValidObjects ) {
if ( oldCertInfo . certInfo . fingerprint256 !== checkCertificateResult . certInfo . fingerprint256 ) {
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , "Resetting sent_history" ) ;
2021-10-27 10:12:18 +02:00
await R . exec ( "DELETE FROM notification_sent_history WHERE type = 'certificate' AND monitor_id = ?" , [
this . id
] ) ;
} else {
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , "No need to reset sent_history" ) ;
log . debug ( "monitor" , oldCertInfo . certInfo . fingerprint256 ) ;
log . debug ( "monitor" , checkCertificateResult . certInfo . fingerprint256 ) ;
2021-10-27 10:12:18 +02:00
}
} else {
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , "Not valid object" ) ;
2021-10-27 10:12:18 +02:00
}
} catch ( e ) { }
2021-10-27 10:03:16 +02:00
}
2022-04-13 18:30:32 +02:00
tlsInfoBean . info _json = JSON . stringify ( checkCertificateResult ) ;
await R . store ( tlsInfoBean ) ;
2021-08-10 11:51:30 +02:00
return checkCertificateResult ;
2021-07-22 10:04:32 +02:00
}
2024-10-06 03:36:54 +02:00
/ * *
* Checks if the monitor is active based on itself and its parents
* @ param { number } monitorID ID of monitor to send
* @ param { boolean } active is active
* @ returns { Promise < boolean > } Is the monitor active ?
* /
static async isActive ( monitorID , active ) {
const parentActive = await Monitor . isParentActive ( monitorID ) ;
return ( active === 1 ) && parentActive ;
}
2022-04-16 22:11:45 +02:00
/ * *
* Send statistics to clients
* @ param { Server } io Socket server instance
* @ param { number } monitorID ID of monitor to send
* @ param { number } userID ID of user to send to
2023-08-11 09:46:41 +02:00
* @ returns { void }
2022-04-16 22:11:45 +02:00
* /
2021-06-30 15:04:58 +02:00
static async sendStats ( io , monitorID , userID ) {
2021-08-30 08:55:33 +02:00
const hasClients = getTotalClientInRoom ( io , userID ) > 0 ;
2023-08-31 23:19:21 +02:00
let uptimeCalculator = await UptimeCalculator . getUptimeCalculator ( monitorID ) ;
2021-08-30 08:55:33 +02:00
if ( hasClients ) {
2023-08-31 23:19:21 +02:00
// Send 24 hour average ping
let data24h = await uptimeCalculator . get24Hour ( ) ;
2023-10-08 20:33:32 +02:00
io . to ( userID ) . emit ( "avgPing" , monitorID , ( data24h . avgPing ) ? Number ( data24h . avgPing . toFixed ( 2 ) ) : null ) ;
2023-08-31 23:19:21 +02:00
// Send 24 hour uptime
io . to ( userID ) . emit ( "uptime" , monitorID , 24 , data24h . uptime ) ;
// Send 30 day uptime
let data30d = await uptimeCalculator . get30Day ( ) ;
io . to ( userID ) . emit ( "uptime" , monitorID , 720 , data30d . uptime ) ;
// Send 1-year uptime
let data1y = await uptimeCalculator . get1Year ( ) ;
io . to ( userID ) . emit ( "uptime" , monitorID , "1y" , data1y . uptime ) ;
// Send Cert Info
2021-08-30 08:55:33 +02:00
await Monitor . sendCertInfo ( io , monitorID , userID ) ;
} else {
2022-04-13 17:33:37 +02:00
log . debug ( "monitor" , "No clients in the room, no need to send stats" ) ;
2021-08-30 08:55:33 +02:00
}
2021-06-30 15:04:58 +02:00
}
2022-04-16 22:11:45 +02:00
/ * *
2022-04-22 19:42:47 +02:00
* Send certificate information to client
2022-04-16 22:11:45 +02:00
* @ param { Server } io Socket server instance
* @ param { number } monitorID ID of monitor to send
* @ param { number } userID ID of user to send to
2023-08-11 09:46:41 +02:00
* @ returns { void }
2022-04-16 22:11:45 +02:00
* /
2021-07-22 10:04:32 +02:00
static async sendCertInfo ( io , monitorID , userID ) {
2022-04-16 19:39:49 +02:00
let tlsInfo = await R . findOne ( "monitor_tls_info" , "monitor_id = ?" , [
2021-07-30 13:18:26 +02:00
monitorID ,
2021-07-22 10:04:32 +02:00
] ) ;
2022-04-16 19:39:49 +02:00
if ( tlsInfo != null ) {
io . to ( userID ) . emit ( "certInfo" , monitorID , tlsInfo . info _json ) ;
2021-07-22 10:04:32 +02:00
}
2021-07-21 06:09:09 +02:00
}
2022-04-16 22:11:45 +02:00
/ * *
* Has status of monitor changed since last beat ?
* @ param { boolean } isFirstBeat Is this the first beat of this monitor ?
* @ param { const } previousBeatStatus Status of the previous beat
* @ param { const } currentBeatStatus Status of the current beat
* @ returns { boolean } True if is an important beat else false
* /
2021-10-14 16:32:15 +02:00
static isImportantBeat ( isFirstBeat , previousBeatStatus , currentBeatStatus ) {
// * ? -> ANY STATUS = important [isFirstBeat]
// UP -> PENDING = not important
// * UP -> DOWN = important
// UP -> UP = not important
// PENDING -> PENDING = not important
// * PENDING -> DOWN = important
// PENDING -> UP = not important
// DOWN -> PENDING = this case not exists
// DOWN -> DOWN = not important
// * DOWN -> UP = important
2022-01-23 16:22:00 +02:00
// MAINTENANCE -> MAINTENANCE = not important
// * MAINTENANCE -> UP = important
// * MAINTENANCE -> DOWN = important
// * DOWN -> MAINTENANCE = important
// * UP -> MAINTENANCE = important
return isFirstBeat ||
( previousBeatStatus === DOWN && currentBeatStatus === MAINTENANCE ) ||
( previousBeatStatus === UP && currentBeatStatus === MAINTENANCE ) ||
( previousBeatStatus === MAINTENANCE && currentBeatStatus === DOWN ) ||
( previousBeatStatus === MAINTENANCE && currentBeatStatus === UP ) ||
( previousBeatStatus === UP && currentBeatStatus === DOWN ) ||
( previousBeatStatus === DOWN && currentBeatStatus === UP ) ||
( previousBeatStatus === PENDING && currentBeatStatus === DOWN ) ;
}
2022-05-30 15:31:45 +02:00
/ * *
* Is this beat important for notifications ?
* @ param { boolean } isFirstBeat Is this the first beat of this monitor ?
* @ param { const } previousBeatStatus Status of the previous beat
* @ param { const } currentBeatStatus Status of the current beat
* @ returns { boolean } True if is an important beat else false
* /
2022-01-23 16:22:00 +02:00
static isImportantForNotification ( isFirstBeat , previousBeatStatus , currentBeatStatus ) {
// * ? -> ANY STATUS = important [isFirstBeat]
// UP -> PENDING = not important
// * UP -> DOWN = important
// UP -> UP = not important
// PENDING -> PENDING = not important
// * PENDING -> DOWN = important
// PENDING -> UP = not important
// DOWN -> PENDING = this case not exists
// DOWN -> DOWN = not important
// * DOWN -> UP = important
// MAINTENANCE -> MAINTENANCE = not important
// MAINTENANCE -> UP = not important
// * MAINTENANCE -> DOWN = important
// DOWN -> MAINTENANCE = not important
// UP -> MAINTENANCE = not important
return isFirstBeat ||
( previousBeatStatus === MAINTENANCE && currentBeatStatus === DOWN ) ||
2021-10-14 16:32:15 +02:00
( previousBeatStatus === UP && currentBeatStatus === DOWN ) ||
( previousBeatStatus === DOWN && currentBeatStatus === UP ) ||
( previousBeatStatus === PENDING && currentBeatStatus === DOWN ) ;
}
2022-04-16 22:11:45 +02:00
/ * *
* Send a notification about a monitor
* @ param { boolean } isFirstBeat Is this beat the first of this monitor ?
* @ param { Monitor } monitor The monitor to send a notificaton about
* @ param { Bean } bean Status information about monitor
2023-08-11 09:46:41 +02:00
* @ returns { void }
2022-04-16 22:11:45 +02:00
* /
2021-10-14 16:32:15 +02:00
static async sendNotification ( isFirstBeat , monitor , bean ) {
if ( ! isFirstBeat || bean . status === DOWN ) {
2021-10-27 09:33:15 +02:00
const notificationList = await Monitor . getNotificationList ( monitor ) ;
2021-10-14 16:32:15 +02:00
let text ;
if ( bean . status === UP ) {
text = "✅ Up" ;
} else {
text = "🔴 Down" ;
}
let msg = ` [ ${ monitor . name } ] [ ${ text } ] ${ bean . msg } ` ;
for ( let notification of notificationList ) {
try {
2022-12-24 08:23:50 +02:00
const heartbeatJSON = bean . toJSON ( ) ;
2024-10-06 03:36:54 +02:00
const monitorData = [ { id : monitor . id ,
active : monitor . active
} ] ;
const preloadData = await Monitor . preparePreloadData ( monitorData ) ;
2023-04-09 10:01:27 +02:00
// Prevent if the msg is undefined, notifications such as Discord cannot send out.
2022-12-24 08:23:50 +02:00
if ( ! heartbeatJSON [ "msg" ] ) {
2022-12-30 07:46:34 +02:00
heartbeatJSON [ "msg" ] = "N/A" ;
2022-12-24 08:23:50 +02:00
}
2023-04-09 10:01:27 +02:00
// Also provide the time in server timezone
heartbeatJSON [ "timezone" ] = await UptimeKumaServer . getInstance ( ) . getTimezone ( ) ;
heartbeatJSON [ "timezoneOffset" ] = UptimeKumaServer . getInstance ( ) . getTimezoneOffset ( ) ;
heartbeatJSON [ "localDateTime" ] = dayjs . utc ( heartbeatJSON [ "time" ] ) . tz ( heartbeatJSON [ "timezone" ] ) . format ( SQL _DATETIME _FORMAT ) ;
2024-10-06 03:36:54 +02:00
await Notification . send ( JSON . parse ( notification . config ) , msg , monitor . toJSON ( preloadData , false ) , heartbeatJSON ) ;
2021-10-14 16:32:15 +02:00
} catch ( e ) {
2022-04-13 17:33:37 +02:00
log . error ( "monitor" , "Cannot send notification to " + notification . name ) ;
log . error ( "monitor" , e ) ;
2021-10-14 16:32:15 +02:00
}
}
}
}
2022-04-16 22:11:45 +02:00
/ * *
* Get list of notification providers for a given monitor
* @ param { Monitor } monitor Monitor to get notification providers for
2023-08-11 09:46:41 +02:00
* @ returns { Promise < LooseObject < any > [ ] > } List of notifications
2022-04-16 22:11:45 +02:00
* /
2021-10-27 09:33:15 +02:00
static async getNotificationList ( monitor ) {
let notificationList = await R . getAll ( "SELECT notification.* FROM notification, monitor_notification WHERE monitor_id = ? AND monitor_notification.notification_id = notification.id " , [
monitor . id ,
] ) ;
return notificationList ;
}
2022-04-16 22:11:45 +02:00
/ * *
2023-01-12 12:39:36 +02:00
* checks certificate chain for expiring certificates
2023-08-11 09:46:41 +02:00
* @ param { object } tlsInfoObject Information about certificate
* @ returns { void }
2022-04-16 22:11:45 +02:00
* /
2023-01-12 12:39:36 +02:00
async checkCertExpiryNotifications ( tlsInfoObject ) {
2021-10-27 09:33:15 +02:00
if ( tlsInfoObject && tlsInfoObject . certInfo && tlsInfoObject . certInfo . daysRemaining ) {
const notificationList = await Monitor . getNotificationList ( this ) ;
2023-01-12 12:39:36 +02:00
if ( ! notificationList . length > 0 ) {
// fail fast. If no notification is set, all the following checks can be skipped.
log . debug ( "monitor" , "No notification, no need to send cert notification" ) ;
return ;
}
2024-10-09 01:43:44 +02:00
let notifyDays = await setting ( "tlsExpiryNotifyDays" ) ;
2022-05-12 12:18:47 +02:00
if ( notifyDays == null || ! Array . isArray ( notifyDays ) ) {
// Reset Default
2024-10-09 01:43:44 +02:00
await setSetting ( "tlsExpiryNotifyDays" , [ 7 , 14 , 21 ] , "general" ) ;
2022-05-12 12:18:47 +02:00
notifyDays = [ 7 , 14 , 21 ] ;
}
2023-01-12 12:39:36 +02:00
if ( Array . isArray ( notifyDays ) ) {
for ( const targetDays of notifyDays ) {
let certInfo = tlsInfoObject . certInfo ;
while ( certInfo ) {
let subjectCN = certInfo . subject [ "CN" ] ;
2023-10-15 20:20:38 +02:00
if ( rootCertificates . has ( certInfo . fingerprint256 ) ) {
log . debug ( "monitor" , ` Known root cert: ${ certInfo . certType } certificate " ${ subjectCN } " ( ${ certInfo . daysRemaining } days valid) on ${ targetDays } deadline. ` ) ;
break ;
} else if ( certInfo . daysRemaining > targetDays ) {
2023-01-12 12:39:36 +02:00
log . debug ( "monitor" , ` No need to send cert notification for ${ certInfo . certType } certificate " ${ subjectCN } " ( ${ certInfo . daysRemaining } days valid) on ${ targetDays } deadline. ` ) ;
} else {
log . debug ( "monitor" , ` call sendCertNotificationByTargetDays for ${ targetDays } deadline on certificate ${ subjectCN } . ` ) ;
await this . sendCertNotificationByTargetDays ( subjectCN , certInfo . certType , certInfo . daysRemaining , targetDays , notificationList ) ;
}
certInfo = certInfo . issuerCertificate ;
}
2022-05-12 12:18:47 +02:00
}
}
2021-10-27 09:33:15 +02:00
}
}
2022-04-16 22:11:45 +02:00
/ * *
* Send a certificate notification when certificate expires in less
* than target days
2023-01-12 12:39:36 +02:00
* @ param { string } certCN Common Name attribute from the certificate subject
* @ param { string } certType certificate type
* @ param { number } daysRemaining Number of days remaining on certificate
2022-04-16 22:11:45 +02:00
* @ param { number } targetDays Number of days to alert after
2022-04-21 21:02:18 +02:00
* @ param { LooseObject < any > [ ] } notificationList List of notification providers
2022-04-16 22:11:45 +02:00
* @ returns { Promise < void > }
* /
2023-01-12 12:39:36 +02:00
async sendCertNotificationByTargetDays ( certCN , certType , daysRemaining , targetDays , notificationList ) {
2021-10-27 09:33:15 +02:00
2023-05-26 15:38:51 +02:00
let row = await R . getRow ( "SELECT * FROM notification_sent_history WHERE type = ? AND monitor_id = ? AND days <= ?" , [
"certificate" ,
this . id ,
targetDays ,
] ) ;
2021-10-27 09:33:15 +02:00
2023-01-12 12:39:36 +02:00
// Sent already, no need to send again
if ( row ) {
log . debug ( "monitor" , "Sent already, no need to send again" ) ;
2021-10-27 09:33:15 +02:00
return ;
}
2023-01-12 12:39:36 +02:00
let sent = false ;
log . debug ( "monitor" , "Send certificate notification" ) ;
2021-10-27 09:33:15 +02:00
2023-01-12 12:39:36 +02:00
for ( let notification of notificationList ) {
try {
log . debug ( "monitor" , "Sending to " + notification . name ) ;
await Notification . send ( JSON . parse ( notification . config ) , ` [ ${ this . name } ][ ${ this . url } ] ${ certType } certificate ${ certCN } will be expired in ${ daysRemaining } days ` ) ;
sent = true ;
} catch ( e ) {
log . error ( "monitor" , "Cannot send cert notification to " + notification . name ) ;
log . error ( "monitor" , e ) ;
}
}
2021-10-27 09:33:15 +02:00
2023-01-12 12:39:36 +02:00
if ( sent ) {
await R . exec ( "INSERT INTO notification_sent_history (type, monitor_id, days) VALUES(?, ?, ?)" , [
2021-10-27 09:33:15 +02:00
"certificate" ,
this . id ,
targetDays ,
] ) ;
}
}
2021-12-08 08:59:59 +02:00
2022-04-16 22:11:45 +02:00
/ * *
* Get the status of the previous heartbeat
* @ param { number } monitorID ID of monitor to check
2023-08-11 09:46:41 +02:00
* @ returns { Promise < LooseObject < any >> } Previous heartbeat
2022-04-16 22:11:45 +02:00
* /
2021-12-08 08:59:59 +02:00
static async getPreviousHeartbeat ( monitorID ) {
2023-11-24 12:11:36 +02:00
return await R . findOne ( "heartbeat" , " id = (select MAX(id) from heartbeat where monitor_id = ?)" , [
2021-12-08 08:59:59 +02:00
monitorID
] ) ;
}
2022-01-25 20:07:27 +02:00
2022-04-30 15:50:05 +02:00
/ * *
* Check if monitor is under maintenance
* @ param { number } monitorID ID of monitor to check
2023-08-11 09:46:41 +02:00
* @ returns { Promise < boolean > } Is the monitor under maintenance
2022-04-30 15:50:05 +02:00
* /
2022-01-25 20:07:27 +02:00
static async isUnderMaintenance ( monitorID ) {
2023-03-30 22:04:17 +02:00
const maintenanceIDList = await R . getCol ( `
SELECT maintenance _id FROM monitor _maintenance
WHERE monitor _id = ?
` , [ monitorID ]);
for ( const maintenanceID of maintenanceIDList ) {
const maintenance = await UptimeKumaServer . getInstance ( ) . getMaintenance ( maintenanceID ) ;
if ( maintenance && await maintenance . isUnderMaintenance ( ) ) {
return true ;
}
2023-05-31 21:29:20 +02:00
}
2023-05-31 21:19:46 +02:00
2023-05-31 21:29:20 +02:00
const parent = await Monitor . getParent ( monitorID ) ;
if ( parent != null ) {
return await Monitor . isUnderMaintenance ( parent . id ) ;
2023-01-28 15:02:10 +02:00
}
2023-03-30 22:04:17 +02:00
return false ;
2022-01-25 20:07:27 +02:00
}
2022-12-08 17:21:55 +02:00
2023-08-11 09:46:41 +02:00
/ * *
* Make sure monitor interval is between bounds
* @ returns { void }
* @ throws Interval is outside of range
* /
2022-12-08 17:21:55 +02:00
validate ( ) {
if ( this . interval > MAX _INTERVAL _SECOND ) {
throw new Error ( ` Interval cannot be more than ${ MAX _INTERVAL _SECOND } seconds ` ) ;
}
if ( this . interval < MIN _INTERVAL _SECOND ) {
throw new Error ( ` Interval cannot be less than ${ MIN _INTERVAL _SECOND } seconds ` ) ;
}
}
2023-01-28 03:58:03 +02:00
2024-10-06 03:36:54 +02:00
/ * *
* Gets monitor notification of multiple monitor
* @ param { Array } monitorIDs IDs of monitor to get
* @ returns { Promise < LooseObject < any >> } object
* /
static async getMonitorNotification ( monitorIDs ) {
return await R . getAll ( `
SELECT monitor _notification . monitor _id , monitor _notification . notification _id
FROM monitor _notification
2024-10-16 13:25:35 +02:00
WHERE monitor _notification . monitor _id IN ( $ { monitorIDs . map ( ( _ ) => "?" ) . join ( "," ) } )
` , monitorIDs);
2024-10-06 03:36:54 +02:00
}
/ * *
* Gets monitor tags of multiple monitor
* @ param { Array } monitorIDs IDs of monitor to get
* @ returns { Promise < LooseObject < any >> } object
* /
static async getMonitorTag ( monitorIDs ) {
return await R . getAll ( `
2024-10-16 13:25:35 +02:00
SELECT monitor _tag . monitor _id , monitor _tag . tag _id , tag . name , tag . color
2024-10-06 03:36:54 +02:00
FROM monitor _tag
JOIN tag ON monitor _tag . tag _id = tag . id
2024-10-16 13:25:35 +02:00
WHERE monitor _tag . monitor _id IN ( $ { monitorIDs . map ( ( _ ) => "?" ) . join ( "," ) } )
` , monitorIDs);
2024-10-06 03:36:54 +02:00
}
/ * *
* prepare preloaded data for efficient access
* @ param { Array } monitorData IDs & active field of monitor to get
* @ returns { Promise < LooseObject < any >> } object
* /
static async preparePreloadData ( monitorData ) {
const notificationsMap = new Map ( ) ;
const tagsMap = new Map ( ) ;
const maintenanceStatusMap = new Map ( ) ;
const childrenIDsMap = new Map ( ) ;
const activeStatusMap = new Map ( ) ;
const forceInactiveMap = new Map ( ) ;
const pathsMap = new Map ( ) ;
if ( monitorData . length > 0 ) {
const monitorIDs = monitorData . map ( monitor => monitor . id ) ;
const notifications = await Monitor . getMonitorNotification ( monitorIDs ) ;
const tags = await Monitor . getMonitorTag ( monitorIDs ) ;
const maintenanceStatuses = await Promise . all ( monitorData . map ( monitor => Monitor . isUnderMaintenance ( monitor . id ) ) ) ;
const childrenIDs = await Promise . all ( monitorData . map ( monitor => Monitor . getAllChildrenIDs ( monitor . id ) ) ) ;
const activeStatuses = await Promise . all ( monitorData . map ( monitor => Monitor . isActive ( monitor . id , monitor . active ) ) ) ;
const forceInactiveStatuses = await Promise . all ( monitorData . map ( monitor => Monitor . isParentActive ( monitor . id ) ) ) ;
const paths = await Promise . all ( monitorData . map ( monitor => Monitor . getAllPath ( monitor . id , monitor . name ) ) ) ;
notifications . forEach ( row => {
if ( ! notificationsMap . has ( row . monitor _id ) ) {
notificationsMap . set ( row . monitor _id , { } ) ;
}
notificationsMap . get ( row . monitor _id ) [ row . notification _id ] = true ;
} ) ;
tags . forEach ( row => {
if ( ! tagsMap . has ( row . monitor _id ) ) {
tagsMap . set ( row . monitor _id , [ ] ) ;
}
tagsMap . get ( row . monitor _id ) . push ( {
2024-10-16 13:25:35 +02:00
tag _id : row . tag _id ,
2024-10-06 03:36:54 +02:00
name : row . name ,
color : row . color
} ) ;
} ) ;
monitorData . forEach ( ( monitor , index ) => {
maintenanceStatusMap . set ( monitor . id , maintenanceStatuses [ index ] ) ;
} ) ;
monitorData . forEach ( ( monitor , index ) => {
childrenIDsMap . set ( monitor . id , childrenIDs [ index ] ) ;
} ) ;
monitorData . forEach ( ( monitor , index ) => {
activeStatusMap . set ( monitor . id , activeStatuses [ index ] ) ;
} ) ;
monitorData . forEach ( ( monitor , index ) => {
forceInactiveMap . set ( monitor . id , ! forceInactiveStatuses [ index ] ) ;
} ) ;
monitorData . forEach ( ( monitor , index ) => {
pathsMap . set ( monitor . id , paths [ index ] ) ;
} ) ;
}
return {
notifications : notificationsMap ,
tags : tagsMap ,
maintenanceStatus : maintenanceStatusMap ,
childrenIDs : childrenIDsMap ,
activeStatus : activeStatusMap ,
forceInactive : forceInactiveMap ,
paths : pathsMap ,
} ;
}
2023-01-28 03:58:03 +02:00
/ * *
* Gets Parent of the monitor
* @ param { number } monitorID ID of monitor to get
2023-08-11 09:46:41 +02:00
* @ returns { Promise < LooseObject < any >> } Parent
2023-01-28 03:58:03 +02:00
* /
static async getParent ( monitorID ) {
return await R . getRow ( `
SELECT parent . * FROM monitor parent
LEFT JOIN monitor child
ON child . parent = parent . id
WHERE child . id = ?
` , [
monitorID ,
] ) ;
}
/ * *
* Gets all Children of the monitor
* @ param { number } monitorID ID of monitor to get
2023-08-11 09:46:41 +02:00
* @ returns { Promise < LooseObject < any >> } Children
2023-01-28 03:58:03 +02:00
* /
static async getChildren ( monitorID ) {
return await R . getAll ( `
SELECT * FROM monitor
WHERE parent = ?
` , [
monitorID ,
] ) ;
}
/ * *
2024-03-03 11:59:38 +02:00
* Gets the full path
2024-10-06 03:36:54 +02:00
* @ param { number } monitorID ID of the monitor to get
* @ param { string } name of the monitor to get
2024-03-03 11:59:38 +02:00
* @ returns { Promise < string [ ] > } Full path ( includes groups and the name ) of the monitor
2023-01-28 03:58:03 +02:00
* /
2024-10-06 03:36:54 +02:00
static async getAllPath ( monitorID , name ) {
const path = [ name ] ;
2023-01-28 03:58:03 +02:00
if ( this . parent === null ) {
return path ;
}
2024-10-06 03:36:54 +02:00
let parent = await Monitor . getParent ( monitorID ) ;
2023-01-28 03:58:03 +02:00
while ( parent !== null ) {
2024-03-03 11:59:38 +02:00
path . unshift ( parent . name ) ;
2023-01-28 03:58:03 +02:00
parent = await Monitor . getParent ( parent . id ) ;
}
return path ;
}
/ * *
* Gets recursive all child ids
2023-08-11 09:46:41 +02:00
* @ param { number } monitorID ID of the monitor to get
* @ returns { Promise < Array > } IDs of all children
2023-01-28 03:58:03 +02:00
* /
static async getAllChildrenIDs ( monitorID ) {
const childs = await Monitor . getChildren ( monitorID ) ;
if ( childs === null ) {
return [ ] ;
}
let childrenIDs = [ ] ;
for ( const child of childs ) {
childrenIDs . push ( child . id ) ;
childrenIDs = childrenIDs . concat ( await Monitor . getAllChildrenIDs ( child . id ) ) ;
}
return childrenIDs ;
}
2023-01-28 14:39:17 +02:00
2023-06-26 04:44:15 +02:00
/ * *
2024-04-16 16:59:07 +02:00
* Unlinks all children of the group monitor
2023-06-26 04:44:15 +02:00
* @ param { number } groupID ID of group to remove children of
* @ returns { Promise < void > }
* /
static async unlinkAllChildren ( groupID ) {
return await R . exec ( "UPDATE `monitor` SET parent = ? WHERE parent = ? " , [
null , groupID
] ) ;
}
2023-01-28 14:39:17 +02:00
/ * *
2023-08-11 09:46:41 +02:00
* Checks recursive if parent ( ancestors ) are active
* @ param { number } monitorID ID of the monitor to get
* @ returns { Promise < boolean > } Is the parent monitor active ?
* /
2023-01-28 14:39:17 +02:00
static async isParentActive ( monitorID ) {
const parent = await Monitor . getParent ( monitorID ) ;
if ( parent === null ) {
return true ;
}
const parentActive = await Monitor . isParentActive ( parent . id ) ;
return parent . active && parentActive ;
}
2023-12-17 11:21:07 +02:00
/ * *
* Obtains a new Oidc Token
* @ returns { Promise < object > } OAuthProvider client
* /
async makeOidcTokenClientCredentialsRequest ( ) {
log . debug ( "monitor" , ` [ ${ this . name } ] The oauth access-token undefined or expired. Requesting a new token ` ) ;
const oAuthAccessToken = await getOidcTokenClientCredentials ( this . oauth _token _url , this . oauth _client _id , this . oauth _client _secret , this . oauth _scopes , this . oauth _auth _method ) ;
if ( this . oauthAccessToken ? . expires _at ) {
log . debug ( "monitor" , ` [ ${ this . name } ] Obtained oauth access-token. Expires at ${ new Date ( this . oauthAccessToken ? . expires _at * 1000 ) } ` ) ;
} else {
log . debug ( "monitor" , ` [ ${ this . name } ] Obtained oauth access-token. Time until expiry was not provided ` ) ;
}
return oAuthAccessToken ;
}
2024-04-24 08:37:17 +02:00
/ * *
* Store TLS certificate information and check for expiry
2024-04-25 17:45:24 +02:00
* @ param { object } tlsInfo Information about the TLS connection
2024-04-24 08:37:17 +02:00
* @ returns { Promise < void > }
* /
async handleTlsInfo ( tlsInfo ) {
await this . updateTlsInfo ( tlsInfo ) ;
this . prometheus ? . update ( null , tlsInfo ) ;
if ( ! this . getIgnoreTls ( ) && this . isEnabledExpiryNotification ( ) ) {
log . debug ( "monitor" , ` [ ${ this . name } ] call checkCertExpiryNotifications ` ) ;
await this . checkCertExpiryNotifications ( tlsInfo ) ;
}
}
2021-06-25 15:55:49 +02:00
}
module . exports = Monitor ;