/** * bf-trigger-camera-changed — fires when a camera entity is created, updated, * or deleted in admin. * * Topic filter: `camera.changed`. Server's nodered-bridge POSTs to * `${noderedUrl}/in/camera.changed` directly. This node self-registers its * own POST handler — no upstream `http in` node required. * * Optional config: * - camera_id: only fire for that camera id * * Output msg.payload: { camera_id, event: "created" | "updated" | "deleted" } */ const { readJsonBody } = require("./_http-body.js"); module.exports = function (RED) { const TOPIC = "camera.changed"; const ROUTE = "/api/internal/" + TOPIC; function BfTriggerCameraChangedNode(config) { RED.nodes.createNode(this, config); const node = this; const filterIdRaw = (config.camera_id || "").toString().trim(); const filterId = filterIdRaw && !isNaN(Number(filterIdRaw)) ? Number(filterIdRaw) : null; async function handler(req, res) { const body = await readJsonBody(req); const camId = body.camera_id !== undefined ? body.camera_id : null; if (filterId !== null && Number(camId) !== filterId) { return res.status(200).end(); } const out = { topic: TOPIC, payload: { camera_id: camId, event: body.event || null, }, }; node.status({ fill: "green", shape: "dot", text: String(out.payload.camera_id || "") + " " + (out.payload.event || ""), }); node.send(out); res.status(200).end(); } RED.httpNode.post(ROUTE, handler); node.on("close", function (done) { const stack = RED.httpNode && RED.httpNode._router && RED.httpNode._router.stack; if (stack) { for (let i = stack.length - 1; i >= 0; i--) { const layer = stack[i]; if (!layer || !layer.route || layer.route.path !== ROUTE) continue; const inner = layer.route.stack; if (Array.isArray(inner)) { for (let j = inner.length - 1; j >= 0; j--) { if (inner[j] && inner[j].handle === handler) inner.splice(j, 1); } if (inner.length === 0) stack.splice(i, 1); } } } done(); }); } RED.nodes.registerType("bf-trigger-camera-changed", BfTriggerCameraChangedNode); };