Adds an api to get asynchronous update events.

This commit is contained in:
2023-09-26 10:50:54 +02:00
parent fdf10196a0
commit 9f7e1cf7ba
3 changed files with 102 additions and 1 deletions

View File

@@ -0,0 +1,73 @@
import type { ServiceData, ServicePoller } from '$lib/services/service';
import { getServiceRecord } from '$lib/services/services';
import { readable, type Readable } from 'svelte/store';
import { serverConfig } from './config';
type MayAsyncServiceData = ServiceData | Promise<ServiceData>;
let serviceData: MayAsyncServiceData[][] | undefined = undefined;
function forEachService(
fn: (group: number, item: number, pollResult: Promise<ServiceData>) => void
): MayAsyncServiceData[][] {
const config = serverConfig();
if (serviceData == undefined) {
serviceData = [];
}
for (const [idxGroup, group] of config.services.entries()) {
if (serviceData[group] == undefined) {
serviceData[group] = [];
}
for (const [idxItem, service] of group.items.entries()) {
const poller: ServicePoller = getServiceRecord(service.type || '').poll;
fn(idxGroup, idxItem, poller(service));
}
}
return serviceData;
}
export function getCurrentServiceData(): MayAsyncServiceData[][] {
if (serviceData != undefined) {
return serviceData;
}
return forEachService((group: number, item: number, pollResult: Promise<ServiceData>) => {
(serviceData as MayAsyncServiceData[][])[group][item] = pollResult.then(
(data: ServiceData) => {
(serviceData as MayAsyncServiceData[][])[group][item] = data;
return data;
}
);
});
}
export interface ServiceDataEvent {
group: number;
item: number;
data: ServiceData;
}
const pollInterval = 30000;
export const serviceDataEvents = readable<ServiceDataEvent>(undefined, (set) => {
console.debug('starting ServiceData polling loop every ' + pollInterval / 1000 + 's');
const timer = setInterval(() => {
forEachService((group: number, item: number, result: Promise<ServiceData>) => {
(serviceData as MayAsyncServiceData[][])[group][item] = result.then(
(data: ServiceData) => {
(serviceData as MayAsyncServiceData[][])[group][item] = data;
set({ group, item, data });
return data;
}
);
});
}, pollInterval);
return () => {
console.debug('stopping ServiceData polling loop');
clearInterval(timer);
};
});

View File

@@ -0,0 +1,27 @@
import { serviceDataEvents } from '$lib/server/dataPolling';
import type { RequestHandler } from '@sveltejs/kit';
import type { Unsubscriber } from 'svelte/store';
export const GET: RequestHandler = () => {
let unsubscribe: Unsubscriber | undefined = undefined;
const stream = new ReadableStream({
start: (controller) => {
console.debug('ServiceDataEvent stream started');
unsubscribe = serviceDataEvents.subscribe((event) => {
const data = `event:message\ndata:${JSON.stringify(event)}\n\n`;
controller.enqueue(data);
});
},
cancel: (reason) => {
console.debug('ServiceDataEvent stream canceled: ', reason);
if (unsubscribe != undefined) {
unsubscribe();
}
}
});
return new Response(stream, {
headers: {
'Content-Type': 'text/event-stream'
}
});
};

View File

@@ -8,5 +8,6 @@ export default defineConfig({
include: ['src/**/*.{test,spec}.{js,ts}']
},
build: { target: 'esnext' }
build: { target: 'esnext' },
logLevel: 'info'
});