2021-02-16 05:10:20 +01:00
|
|
|
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({
|
2021-02-18 05:53:57 +01:00
|
|
|
config,
|
2021-02-16 05:10:20 +01:00
|
|
|
children,
|
|
|
|
createWebsocket = defaultCreateWebsocket,
|
2021-02-20 15:11:58 +01:00
|
|
|
mqttUrl = `${baseUrl.replace(/^http/, 'ws')}/ws`,
|
2021-02-16 05:10:20 +01:00
|
|
|
}) {
|
|
|
|
const [state, dispatch] = useReducer(reducer, initialState);
|
|
|
|
const wsRef = useRef();
|
|
|
|
|
2021-02-18 05:53:57 +01:00
|
|
|
useEffect(() => {
|
|
|
|
Object.keys(config.cameras).forEach((camera) => {
|
2021-07-11 21:49:10 +02:00
|
|
|
const { name, record, detect, snapshots } = config.cameras[camera];
|
|
|
|
dispatch({ topic: `${name}/recordings/state`, payload: record.enabled ? 'ON' : 'OFF' });
|
2021-02-18 05:53:57 +01:00
|
|
|
dispatch({ topic: `${name}/detect/state`, payload: detect.enabled ? 'ON' : 'OFF' });
|
|
|
|
dispatch({ topic: `${name}/snapshots/state`, payload: snapshots.enabled ? 'ON' : 'OFF' });
|
|
|
|
});
|
|
|
|
}, [config]);
|
|
|
|
|
2021-02-16 05:10:20 +01:00
|
|
|
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 <Mqtt.Provider value={{ state, ws: wsRef.current }}>{children}</Mqtt.Provider>;
|
|
|
|
}
|
|
|
|
|
2021-07-06 14:22:48 +02:00
|
|
|
export function useMqtt(watchTopic, publishTopic) {
|
2021-02-16 05:10:20 +01:00
|
|
|
const { state, ws } = useContext(Mqtt);
|
|
|
|
|
2021-06-23 15:48:56 +02:00
|
|
|
const value = state[watchTopic] || { payload: null };
|
2021-02-16 05:10:20 +01:00
|
|
|
|
|
|
|
const send = useCallback(
|
|
|
|
(payload) => {
|
2021-02-18 05:53:57 +01:00
|
|
|
ws.send(
|
|
|
|
JSON.stringify({
|
|
|
|
topic: publishTopic || watchTopic,
|
|
|
|
payload: typeof payload !== 'string' ? JSON.stringify(payload) : payload,
|
|
|
|
})
|
|
|
|
);
|
2021-02-16 05:10:20 +01:00
|
|
|
},
|
2021-07-06 14:22:48 +02:00
|
|
|
[ws, watchTopic, publishTopic]
|
2021-02-16 05:10:20 +01:00
|
|
|
);
|
|
|
|
|
|
|
|
return { value, send, connected: state.__connected };
|
|
|
|
}
|
2021-02-18 05:53:57 +01:00
|
|
|
|
|
|
|
export function useDetectState(camera) {
|
|
|
|
const {
|
|
|
|
value: { payload },
|
|
|
|
send,
|
|
|
|
connected,
|
|
|
|
} = useMqtt(`${camera}/detect/state`, `${camera}/detect/set`);
|
|
|
|
return { payload, send, connected };
|
|
|
|
}
|
|
|
|
|
2021-07-11 21:49:10 +02:00
|
|
|
export function useRecordingsState(camera) {
|
2021-02-18 05:53:57 +01:00
|
|
|
const {
|
|
|
|
value: { payload },
|
|
|
|
send,
|
|
|
|
connected,
|
2021-07-11 21:49:10 +02:00
|
|
|
} = useMqtt(`${camera}/recordings/state`, `${camera}/recordings/set`);
|
2021-02-18 05:53:57 +01:00
|
|
|
return { payload, send, connected };
|
|
|
|
}
|
|
|
|
|
|
|
|
export function useSnapshotsState(camera) {
|
|
|
|
const {
|
|
|
|
value: { payload },
|
|
|
|
send,
|
|
|
|
connected,
|
|
|
|
} = useMqtt(`${camera}/snapshots/state`, `${camera}/snapshots/set`);
|
|
|
|
return { payload, send, connected };
|
|
|
|
}
|
2021-06-21 17:07:14 +02:00
|
|
|
|
|
|
|
export function useRestart() {
|
|
|
|
const {
|
2021-07-06 14:22:48 +02:00
|
|
|
value: { payload },
|
2021-06-21 17:07:14 +02:00
|
|
|
send,
|
|
|
|
connected,
|
2021-07-06 14:22:48 +02:00
|
|
|
} = useMqtt('restart', 'restart');
|
|
|
|
return { payload, send, connected };
|
2021-06-21 17:07:14 +02:00
|
|
|
}
|