mirror of
https://github.com/blakeblackshear/frigate.git
synced 2024-11-21 19:07:46 +01:00
c3b313a70d
* Initial audio classification model implementation * fix mypy * Keep audio labelmap local * Cleanup * Start adding config for audio * Add the detector * Add audio detection process keypoints * Build out base config * Load labelmap correctly * Fix config bugs * Start audio process * Fix startup issues * Try to cleanup restarting * Add ffmpeg input args * Get audio detection working * Save event to db * End events if not heard for 30 seconds * Use not heard config * Stop ffmpeg when shutting down * Fixes * End events correctly * Use api instead of event queue to save audio events * Get events working * Close threads when stop event is sent * remove unused * Only start audio process if at least one camera is enabled * Add const for float * Cleanup labelmap * Add audio icon in frontend * Add ability to toggle audio with mqtt * Set initial audio value * Fix audio enabling * Close logpipe * Isort * Formatting * Fix web tests * Fix web tests * Handle cases where args are a string * Remove log * Cleanup process close * Use correct field * Simplify if statement * Use var for localhost * Add audio detectors docs * Add restream docs to mention audio detection * Add full config docs * Fix links to other docs --------- Co-authored-by: Jason Hunter <hunterjm@gmail.com>
150 lines
4.0 KiB
JavaScript
150 lines
4.0 KiB
JavaScript
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 WS = 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 WsProvider({
|
|
config,
|
|
children,
|
|
createWebsocket = defaultCreateWebsocket,
|
|
wsUrl = `${baseUrl.replace(/^http/, 'ws')}ws`,
|
|
}) {
|
|
const [state, dispatch] = useReducer(reducer, initialState);
|
|
const wsRef = useRef();
|
|
|
|
useEffect(() => {
|
|
Object.keys(config.cameras).forEach((camera) => {
|
|
const { name, record, detect, snapshots, audio } = config.cameras[camera];
|
|
dispatch({ topic: `${name}/recordings/state`, payload: record.enabled ? 'ON' : 'OFF', retain: false });
|
|
dispatch({ topic: `${name}/detect/state`, payload: detect.enabled ? 'ON' : 'OFF', retain: false });
|
|
dispatch({ topic: `${name}/snapshots/state`, payload: snapshots.enabled ? 'ON' : 'OFF', retain: false });
|
|
dispatch({ topic: `${name}/audio/state`, payload: audio.enabled ? 'ON' : 'OFF', retain: false });
|
|
});
|
|
}, [config]);
|
|
|
|
useEffect(
|
|
() => {
|
|
const ws = createWebsocket(wsUrl);
|
|
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, wsUrl] // eslint-disable-line react-hooks/exhaustive-deps
|
|
);
|
|
|
|
return <WS.Provider value={{ state, ws: wsRef.current }}>{children}</WS.Provider>;
|
|
}
|
|
|
|
export function useWs(watchTopic, publishTopic) {
|
|
const { state, ws } = useContext(WS);
|
|
|
|
const value = state[watchTopic] || { payload: null };
|
|
|
|
const send = useCallback(
|
|
(payload, retain = false) => {
|
|
ws.send(
|
|
JSON.stringify({
|
|
topic: publishTopic || watchTopic,
|
|
payload: typeof payload !== 'string' ? JSON.stringify(payload) : payload,
|
|
retain,
|
|
})
|
|
);
|
|
},
|
|
[ws, watchTopic, publishTopic]
|
|
);
|
|
|
|
return { value, send, connected: state.__connected };
|
|
}
|
|
|
|
export function useDetectState(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/detect/state`, `${camera}/detect/set`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function useRecordingsState(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/recordings/state`, `${camera}/recordings/set`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function useSnapshotsState(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/snapshots/state`, `${camera}/snapshots/set`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function useAudioState(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/audio/state`, `${camera}/audio/set`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function usePtzCommand(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/ptz`, `${camera}/ptz`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function useRestart() {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs('restart', 'restart');
|
|
return { payload, send, connected };
|
|
}
|