import { h, createContext } from 'preact'; import { baseUrl } from './baseUrl'; import produce from 'immer'; import { useCallback, useContext, useEffect, useRef, useReducer } from 'preact/hooks'; const initialState = Object.freeze({ __connected: false }); export const Mqtt = createContext({ state: initialState, connection: null }); const defaultCreateWebsocket = (url) => new WebSocket(url); function reducer(state, { topic, payload, retain }) { switch (topic) { case '__CLIENT_CONNECTED': return produce(state, (draftState) => { draftState.__connected = true; }); default: return produce(state, (draftState) => { let parsedPayload = payload; try { parsedPayload = payload && JSON.parse(payload); } catch (e) {} draftState[topic] = { lastUpdate: Date.now(), payload: parsedPayload, retain, }; }); } } export function MqttProvider({ config, children, createWebsocket = defaultCreateWebsocket, mqttUrl = `${baseUrl.replace(/^http/, 'ws')}/ws`, }) { const [state, dispatch] = useReducer(reducer, initialState); const wsRef = useRef(); useEffect(() => { Object.keys(config.cameras).forEach((camera) => { const { name, clips, detect, snapshots } = config.cameras[camera]; dispatch({ topic: `${name}/clips/state`, payload: clips.enabled ? 'ON' : 'OFF' }); dispatch({ topic: `${name}/detect/state`, payload: detect.enabled ? 'ON' : 'OFF' }); dispatch({ topic: `${name}/snapshots/state`, payload: snapshots.enabled ? 'ON' : 'OFF' }); }); }, [config]); useEffect( () => { const ws = createWebsocket(mqttUrl); ws.onopen = () => { dispatch({ topic: '__CLIENT_CONNECTED' }); }; ws.onmessage = (event) => { dispatch(JSON.parse(event.data)); }; wsRef.current = ws; return () => { ws.close(3000, 'Provider destroyed'); }; }, // Forces reconnecting [state.__reconnectAttempts, mqttUrl] // eslint-disable-line react-hooks/exhaustive-deps ); return {children}; } export function useMqtt(watchTopic, publishTopic, defaultValue = null) { const { state, ws } = useContext(Mqtt); const value = state[watchTopic] || { payload: defaultValue }; const send = useCallback( (payload) => { payload = payload || defaultValue; ws.send( JSON.stringify({ topic: publishTopic || watchTopic, payload: typeof payload !== 'string' ? JSON.stringify(payload) : payload, }) ); }, [ws, watchTopic, publishTopic] ); return { value, send, connected: state.__connected }; } export function useDetectState(camera) { const { value: { payload }, send, connected, } = useMqtt(`${camera}/detect/state`, `${camera}/detect/set`); return { payload, send, connected }; } export function useClipsState(camera) { const { value: { payload }, send, connected, } = useMqtt(`${camera}/clips/state`, `${camera}/clips/set`); return { payload, send, connected }; } export function useSnapshotsState(camera) { const { value: { payload }, send, connected, } = useMqtt(`${camera}/snapshots/state`, `${camera}/snapshots/set`); return { payload, send, connected }; } export function useRestart() { const { value: { payload }, send, connected, } = useMqtt(``, `restart`, "container"); return { send, connected }; }