bb01e9a06c
## Summary - **ADS-B & AIS:** OpenSky and AISStream OSINT feeds upsert into the CoT store; tactical tracks still arrive via adsbcot/aiscot on `:8089`. Map clients subscribe via `GET /api/cot/stream` (SSE) with viewport bbox filtering and Air / Surface / Team layer toggles. - **ALPR (Flock/OSM):** Toggleable license-plate reader layer sourced from OpenStreetMap, with SQLite cache, Overpass fallback, tiled viewport fetching, and clustered markers with direction cones. - **Map performance:** Ring-based tile selection (fixes zoom-out crash), immutable tile cache, incremental marker sync, split cluster load/query, and padded SSE bbox to reduce reconnect churn. ## Docs - `docs/tracking.md` — ADS-B/AIS accuracy tiers, freshness, self-hosted receivers, optional OSINT API keys - `docs/map-and-cameras.md` — ALPR layer and map behavior updates --------- Co-authored-by: Madison Grubb <madison@elastiflow.com> Reviewed-on: #36
83 lines
2.3 KiB
JavaScript
83 lines
2.3 KiB
JavaScript
import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest'
|
|
import { SHUTDOWN_TIMEOUT_MS } from '../../server/utils/constants.js'
|
|
import { registerCleanup, graceful, initShutdownHandlers, clearCleanup } from '../../server/utils/shutdown.js'
|
|
|
|
describe('shutdown integration', () => {
|
|
/** @type {import('vitest').MockInstance} */
|
|
let exitSpy
|
|
/** @type {typeof process.on} */
|
|
let originalOn
|
|
|
|
beforeEach(() => {
|
|
clearCleanup()
|
|
exitSpy = vi.spyOn(process, 'exit').mockImplementation(() => {})
|
|
originalOn = process.on
|
|
})
|
|
|
|
afterEach(() => {
|
|
exitSpy.mockRestore()
|
|
process.on = originalOn
|
|
clearCleanup()
|
|
vi.useRealTimers()
|
|
})
|
|
|
|
it('initializes signal handlers', () => {
|
|
const handlers = {}
|
|
process.on = vi.fn((signal, handler) => {
|
|
handlers[signal] = handler
|
|
})
|
|
initShutdownHandlers()
|
|
expect(process.on).toHaveBeenCalledWith('SIGTERM', expect.any(Function))
|
|
expect(process.on).toHaveBeenCalledWith('SIGINT', expect.any(Function))
|
|
})
|
|
|
|
it('signal handler calls graceful', async () => {
|
|
const handlers = {}
|
|
process.on = vi.fn((signal, handler) => {
|
|
handlers[signal] = handler
|
|
})
|
|
initShutdownHandlers()
|
|
const sigtermHandler = handlers.SIGTERM
|
|
expect(sigtermHandler).toBeDefined()
|
|
sigtermHandler()
|
|
await vi.waitFor(() => {
|
|
expect(exitSpy).toHaveBeenCalled()
|
|
})
|
|
})
|
|
|
|
it('signal handler handles graceful error', async () => {
|
|
const handlers = {}
|
|
process.on = vi.fn((signal, handler) => {
|
|
handlers[signal] = handler
|
|
})
|
|
initShutdownHandlers()
|
|
const sigintHandler = handlers.SIGINT
|
|
vi.spyOn(console, 'error').mockImplementation(() => {})
|
|
registerCleanup(async () => {
|
|
throw new Error('Force error')
|
|
})
|
|
sigintHandler()
|
|
await vi.waitFor(() => {
|
|
expect(exitSpy).toHaveBeenCalled()
|
|
})
|
|
})
|
|
|
|
it('covers timeout path in graceful', async () => {
|
|
vi.useFakeTimers()
|
|
registerCleanup(async () => {
|
|
await new Promise(resolve => setTimeout(resolve, SHUTDOWN_TIMEOUT_MS + 5_000))
|
|
})
|
|
graceful()
|
|
await vi.advanceTimersByTimeAsync(SHUTDOWN_TIMEOUT_MS + 1)
|
|
expect(exitSpy).toHaveBeenCalledWith(1)
|
|
})
|
|
|
|
it('covers graceful catch block', async () => {
|
|
registerCleanup(async () => {
|
|
throw new Error('Test error')
|
|
})
|
|
await graceful()
|
|
expect(exitSpy).toHaveBeenCalled()
|
|
})
|
|
})
|