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
97 lines
3.1 KiB
JavaScript
97 lines
3.1 KiB
JavaScript
import { describe, it, expect, beforeEach, vi } from 'vitest'
|
|
import {
|
|
registerSubscriber,
|
|
getSubscriberBboxUnion,
|
|
clearSubscribers,
|
|
notifySubscribersForEntity,
|
|
notifySubscribersRemove,
|
|
broadcastSubscriberSnapshots,
|
|
} from '../../../server/utils/cotSubscribers.js'
|
|
import { updateFromCot, clearCotStore } from '../../../server/utils/cotStore.js'
|
|
|
|
describe('cotSubscribers', () => {
|
|
beforeEach(() => {
|
|
clearSubscribers()
|
|
})
|
|
|
|
it('unions subscriber bboxes', () => {
|
|
registerSubscriber({
|
|
bbox: { west: -123, south: 37, east: -122, north: 38 },
|
|
layers: new Set(['air']),
|
|
push: vi.fn(),
|
|
})
|
|
registerSubscriber({
|
|
bbox: { west: -124, south: 36, east: -121, north: 39 },
|
|
layers: new Set(['surface']),
|
|
push: vi.fn(),
|
|
})
|
|
expect(getSubscriberBboxUnion()).toEqual({ west: -124, south: 36, east: -121, north: 39 })
|
|
})
|
|
|
|
it('notifies subscribers inside bbox and matching layer', async () => {
|
|
const push = vi.fn()
|
|
registerSubscriber({
|
|
bbox: { west: -123, south: 37, east: -122, north: 38 },
|
|
layers: new Set(['air']),
|
|
push,
|
|
})
|
|
await notifySubscribersForEntity('update', { entity: { id: 'ICAO.x' } }, {
|
|
id: 'ICAO.x',
|
|
lat: 37.5,
|
|
lng: -122.5,
|
|
type: 'a-f-A-C-F',
|
|
})
|
|
expect(push).toHaveBeenCalledWith('update', expect.any(String))
|
|
})
|
|
|
|
it('skips subscribers when entity outside bbox', async () => {
|
|
const push = vi.fn()
|
|
registerSubscriber({
|
|
bbox: { west: -123, south: 37, east: -122, north: 38 },
|
|
layers: new Set(['air']),
|
|
push,
|
|
})
|
|
await notifySubscribersForEntity('update', { entity: { id: 'ICAO.x' } }, {
|
|
id: 'ICAO.x',
|
|
lat: 40,
|
|
lng: -122.5,
|
|
type: 'a-f-A-C-F',
|
|
})
|
|
expect(push).not.toHaveBeenCalled()
|
|
})
|
|
|
|
it('notifySubscribersRemove pushes to all subscribers', async () => {
|
|
const pushA = vi.fn()
|
|
const pushB = vi.fn()
|
|
registerSubscriber({
|
|
bbox: { west: -123, south: 37, east: -122, north: 38 },
|
|
layers: new Set(['air']),
|
|
push: pushA,
|
|
})
|
|
registerSubscriber({
|
|
bbox: { west: -125, south: 35, east: -120, north: 40 },
|
|
layers: new Set(['surface']),
|
|
push: pushB,
|
|
})
|
|
await notifySubscribersRemove('ICAO.removed')
|
|
expect(pushA).toHaveBeenCalledWith('remove', JSON.stringify({ id: 'ICAO.removed' }))
|
|
expect(pushB).toHaveBeenCalledWith('remove', JSON.stringify({ id: 'ICAO.removed' }))
|
|
})
|
|
|
|
it('broadcastSubscriberSnapshots sends per-subscriber filtered snapshot', async () => {
|
|
clearCotStore()
|
|
await updateFromCot({ id: 'ICAO.in', lat: 37.5, lng: -122.5, type: 'a-f-A-C-F' })
|
|
await updateFromCot({ id: 'ICAO.out', lat: 40, lng: -100, type: 'a-f-A-C-F' })
|
|
const push = vi.fn()
|
|
registerSubscriber({
|
|
bbox: { west: -123, south: 37, east: -122, north: 38 },
|
|
layers: new Set(['air']),
|
|
push,
|
|
})
|
|
await broadcastSubscriberSnapshots({ ttlMs: 90_000, osintTtlMs: 30_000, takFilterBbox: false })
|
|
expect(push).toHaveBeenCalledWith('snapshot', expect.any(String))
|
|
const payload = JSON.parse(push.mock.calls[0][1])
|
|
expect(payload.entities.map(e => e.id)).toEqual(['ICAO.in'])
|
|
})
|
|
})
|