2020-10-15 22:00:21 +00:00
|
|
|
import { logger } from "@coder/logger"
|
|
|
|
import { promises as fs } from "fs"
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Provides a heartbeat using a local file to indicate activity.
|
|
|
|
*/
|
|
|
|
export class Heart {
|
|
|
|
private heartbeatTimer?: NodeJS.Timeout
|
|
|
|
private heartbeatInterval = 60000
|
|
|
|
public lastHeartbeat = 0
|
|
|
|
|
|
|
|
public constructor(private readonly heartbeatPath: string, private readonly isActive: () => Promise<boolean>) {}
|
|
|
|
|
|
|
|
public alive(): boolean {
|
|
|
|
const now = Date.now()
|
|
|
|
return now - this.lastHeartbeat < this.heartbeatInterval
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Write to the heartbeat file if we haven't already done so within the
|
|
|
|
* timeout and start or reset a timer that keeps running as long as there is
|
|
|
|
* activity. Failures are logged as warnings.
|
|
|
|
*/
|
|
|
|
public beat(): void {
|
2020-10-27 22:48:37 +00:00
|
|
|
if (this.alive()) {
|
|
|
|
return
|
2020-10-15 22:00:21 +00:00
|
|
|
}
|
2020-10-27 22:48:37 +00:00
|
|
|
|
|
|
|
logger.trace("heartbeat")
|
|
|
|
fs.writeFile(this.heartbeatPath, "").catch((error) => {
|
|
|
|
logger.warn(error.message)
|
|
|
|
})
|
|
|
|
this.lastHeartbeat = Date.now()
|
|
|
|
if (typeof this.heartbeatTimer !== "undefined") {
|
|
|
|
clearTimeout(this.heartbeatTimer)
|
|
|
|
}
|
|
|
|
this.heartbeatTimer = setTimeout(() => {
|
|
|
|
this.isActive()
|
|
|
|
.then((active) => {
|
|
|
|
if (active) {
|
|
|
|
this.beat()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.catch((error) => {
|
|
|
|
logger.warn(error.message)
|
|
|
|
})
|
|
|
|
}, this.heartbeatInterval)
|
2020-10-15 22:00:21 +00:00
|
|
|
}
|
|
|
|
}
|