Skip to content

Commit

Permalink
Merge pull request #1123 from appsignal/implement-heartbeat-checkins
Browse files Browse the repository at this point in the history
Implement heartbeat check-ins
  • Loading branch information
unflxw authored Sep 26, 2024
2 parents 8c70151 + a754e8a commit 401d970
Show file tree
Hide file tree
Showing 22 changed files with 550 additions and 129 deletions.
30 changes: 30 additions & 0 deletions .changesets/add-heartbeat-check-ins.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
---
bump: minor
type: add
---

Add support for heartbeat check-ins.

Use the `checkIn.heartbeat` method to send a single heartbeat check-in event from your application. This can be used, for example, in your application's main loop:

```js
import { checkIn } from "@appsignal/nodejs"

while (true) {
checkIn.heartbeat("job_processor")
await processJob()
}
```

Heartbeats are deduplicated and sent asynchronously, without blocking the current thread. Regardless of how often the `.heartbeat` method is called, at most one heartbeat with the same identifier will be sent every ten seconds.

Pass `{continuous: true}` as the second argument to send heartbeats continuously during the entire lifetime of the current process. This can be used, for example, after your application has finished its boot process:

```js
import { checkIn } from "@appsignal/nodejs"

function main() {
checkIn.heartbeat("job_processor", {continuous: true})
startApp()
}
```
105 changes: 77 additions & 28 deletions src/__tests__/check_in.test.ts
Original file line number Diff line number Diff line change
@@ -1,21 +1,20 @@
import nock from "nock"
import { cron, Cron, Event, EventKind } from "../check_in"
import {
scheduler,
resetScheduler,
setDebounceTime,
resetDebounceTime,
debounceTime
} from "../check_in/scheduler"
import { cron, Cron, heartbeat } from "../check_in"
import { Event, EventKind } from "../check_in/event"
import { scheduler, resetScheduler, debounceTime } from "../check_in/scheduler"
import { Client, Options } from "../client"
import {
heartbeat,
heartbeat as deprecatedHeartbeat,
Heartbeat,
heartbeatClassWarnOnce,
heartbeatHelperWarnOnce
} from "../heartbeat"
import { ndjsonParse } from "../utils"
import type { InternalLogger } from "../internal_logger"
import {
heartbeatInterval,
killContinuousHeartbeats
} from "../check_in/heartbeat"

const DEFAULT_CLIENT_CONFIG: Partial<Options> = {
active: true,
Expand Down Expand Up @@ -107,7 +106,7 @@ function spyOnInternalLogger(
return spies as Required<typeof spies>
}

describe("checkIn.Cron", () => {
describe("Check-ins", () => {
let client: Client
let theCron: Cron
let requests: Request[]
Expand All @@ -122,7 +121,7 @@ describe("checkIn.Cron", () => {

beforeEach(async () => {
await resetScheduler()
resetDebounceTime()
debounceTime.reset()

client = new Client(DEFAULT_CLIENT_CONFIG)

Expand All @@ -137,7 +136,7 @@ describe("checkIn.Cron", () => {

afterAll(async () => {
await resetScheduler()
resetDebounceTime()
debounceTime.reset()
nock.restore()
})

Expand Down Expand Up @@ -229,7 +228,7 @@ describe("checkIn.Cron", () => {
// These tests will mock requests one by one, so that
// they can await their responses.
nock.cleanAll()
setDebounceTime(() => 20)
debounceTime.set(() => 20)
})

it("transmits events close to each other in time in a single request", async () => {
Expand Down Expand Up @@ -342,7 +341,7 @@ describe("checkIn.Cron", () => {
// Set a very long debounce time to ensure that the scheduler
// is not awaiting it, but rather sending the events immediately
// on shutdown.
setDebounceTime(() => 10000)
debounceTime.set(() => 10000)

theCron.start()

Expand All @@ -357,29 +356,29 @@ describe("checkIn.Cron", () => {
it("uses the last transmission time to calculate the debounce", async () => {
const request = mockOneCheckInRequest()

const debounceTime = jest.fn(_lastTransmission => 0)
setDebounceTime(debounceTime)
const debounceTimeMock = jest.fn(_lastTransmission => 0)
debounceTime.set(debounceTimeMock)

theCron.start()

expect(debounceTime).toHaveBeenCalledTimes(1)
expect(debounceTime).toHaveBeenLastCalledWith(undefined)
expect(debounceTimeMock).toHaveBeenCalledTimes(1)
expect(debounceTimeMock).toHaveBeenLastCalledWith(undefined)

await request
const expectedLastTransmission = Date.now()

theCron.finish()

expect(debounceTime).toHaveBeenCalledTimes(2)
expect(debounceTime).not.toHaveBeenLastCalledWith(undefined)
expect(debounceTimeMock).toHaveBeenCalledTimes(2)
expect(debounceTimeMock).not.toHaveBeenLastCalledWith(undefined)
// Allow for some margin of error in the timing of the tests.
expect(debounceTime.mock.calls[1][0]).toBeGreaterThan(
expect(debounceTimeMock.mock.calls[1][0]).toBeGreaterThan(
expectedLastTransmission - 20
)
})

describe("debounce time", () => {
beforeEach(resetDebounceTime)
beforeEach(debounceTime.reset)

it("is short when no last transmission time is given", () => {
expect(debounceTime(undefined)).toBe(100)
Expand Down Expand Up @@ -452,7 +451,7 @@ describe("checkIn.Cron", () => {
// This is to ensure that the scheduler will not send the start and
// finish events together at shutdown, rather than in separate
// requests.
setDebounceTime(() => {
debounceTime.set(() => {
return 10000
})

Expand All @@ -477,7 +476,7 @@ describe("checkIn.Cron", () => {
// This is to ensure that the scheduler will not send the start and
// finish events together at shutdown, rather than in separate
// requests.
setDebounceTime(() => 10000)
debounceTime.set(() => 10000)

await expect(
cron("test-cron-checkin", async () => {
Expand All @@ -493,13 +492,63 @@ describe("checkIn.Cron", () => {
})
})

describe("Appsignal.checkIn.heartbeat()", () => {
beforeAll(() => {
heartbeatInterval.set(() => 20)
})

afterEach(() => {
killContinuousHeartbeats()
})

afterAll(() => {
heartbeatInterval.reset()
})

it("sends a heartbeat event", async () => {
heartbeat("test-heartbeat")

await scheduler.shutdown()

expect(requests).toHaveLength(1)
expectEvents(requests[0], [
{
identifier: "test-heartbeat",
check_in_type: "heartbeat"
}
])
})

describe("with the `continuous: true` option", () => {
it("sends heartbeat events continuously", async () => {
debounceTime.set(() => 20)

heartbeat("test-heartbeat", { continuous: true })

await sleep(60)
await scheduler.shutdown()

expect(requests.length).toBeGreaterThanOrEqual(2)

for (const request of requests) {
expectEvents(request, [
{
identifier: "test-heartbeat",
check_in_type: "heartbeat"
}
])
}
})
})
})

describe("Appsignal.heartbeat (deprecated)", () => {
beforeEach(() => {
heartbeatHelperWarnOnce.reset()
})

it("behaves like Appsignal.checkIn.cron", async () => {
expect(heartbeat("test-cron-checkin")).toBeUndefined()
expect(deprecatedHeartbeat("test-cron-checkin")).toBeUndefined()

await scheduler.shutdown()

Expand All @@ -513,7 +562,7 @@ describe("checkIn.Cron", () => {
.spyOn(Client.internalLogger, "warn")
.mockImplementation()

expect(heartbeat("test-cron-checkin")).toBeUndefined()
expect(deprecatedHeartbeat("test-cron-checkin")).toBeUndefined()

for (const spy of [consoleWarnSpy, internalLoggerWarnSpy]) {
expect(spy.mock.calls).toHaveLength(1)
Expand All @@ -529,7 +578,7 @@ describe("checkIn.Cron", () => {
.spyOn(Client.internalLogger, "warn")
.mockImplementation()

expect(heartbeat("test-cron-checkin")).toBeUndefined()
expect(deprecatedHeartbeat("test-cron-checkin")).toBeUndefined()

for (const spy of [consoleWarnSpy, internalLoggerWarnSpy]) {
expect(spy.mock.calls).toHaveLength(1)
Expand All @@ -539,7 +588,7 @@ describe("checkIn.Cron", () => {
spy.mockClear()
}

expect(heartbeat("test-cron-checkin")).toBeUndefined()
expect(deprecatedHeartbeat("test-cron-checkin")).toBeUndefined()

for (const spy of [consoleWarnSpy, internalLoggerWarnSpy]) {
expect(spy.mock.calls).toHaveLength(0)
Expand Down
Loading

0 comments on commit 401d970

Please sign in to comment.