diff --git a/flows.json b/flows.json index 585481e..acbce37 100644 --- a/flows.json +++ b/flows.json @@ -2431,7 +2431,10 @@ "334a79e6f5f3a16a", "2718f2695ff412fa", "7d50b11bbf5de8fd", - "53b440a31d30df1b" + "53b440a31d30df1b", + "f1f2ac113fe2b639", + "b96ad84b569ac360", + "894e510fd44b01e0" ], "x": 14, "y": 1319, @@ -23974,7 +23977,7 @@ "z": "72f99805df043603", "g": "5802ea32c1de36e8", "name": "Processing", - "func": "node.log(\"Emma Bedroom Climate: Processing Started\");\n// pull in the necessary information\n\nconst states = global.get('homeassistant.homeAssistant.states')\nconst allowed = states['input_boolean.emma_bedroom_climate_protocol'].state\nconst ac = global.get('emmaBedroom.aircon.installed', \"diskCon\")\nconst temp = global.get(\"outdoorTemp.tempStr\")\nconst payload = msg.payload\nconst vacation = states['input_boolean.vacation_mode'].state\nconst dayTemp = states['input_number.emma_bedroom_daytime_temp'].state\nconst nightTemp = states['input_number.emma_bedroom_night_temp'].state\nconst bedTemp = states['input_number.emma_bedroom_bedtime_temp'].state\nconst schedMode = states['input_select.scheduled_climate_mode_emma_bedroom'].state\nconst sleeping = states['input_boolean.emma_sleeping'].state\nconst hotDay = states['input_boolean.hot_day'].state\nconst heatWarning = states[\"binary_sensor.heat_warning\"].state\nconst earlyNight = states[\"binary_sensor.early_night_mode\"].state\nconst danger = states['binary_sensor.heat_warning'].attributes.danger\nconst meltdown = states['input_boolean.meltdown_protocol'].state\nconst coolingActive = states['input_boolean.emma_bedroom_cooling_on'].state\nnode.log(\"Emma Bedroom Climate: Constants Set\")\n\n// Define reusable constants\n\nconst airconEntity = [\"climate.emma_bedroom_aircon\"]\nconst whiteNoiseEntity = [\"input_boolean.white_noise_emma_bedroom\"]\nconst validSchedModesAC = ['AC', 'Fan']\n\n// Helper function to convert a string to title case\n\nfunction convertToTitleCase(str) {\n if (!str) {\n return \"\";\n }\n return str.toLowerCase().replace(/\\b\\w/g, (s) => s.toUpperCase());\n}\n\n// init variables\n\nlet setTemp = []\nlet setEco = []\nlet setHvac = []\nlet setCool = []\nlet setSleep = []\nlet setPeople = []\nlet setDisplay = []\nlet setWhiteNoise = \"turn_off\"\nlet time = []\nlet type = msg.type\nlet topic = msg.topic\nlet isWakeup = context.get(\"isWakeup\")\nnode.log(\"Emma Bedroom Climate: Variables Defined\")\n\n// Sleep Switch Handling\nif (type === 'sleep' && payload === 'off') {\n setDisplay = 'turn_on'\n setPeople = 'turn_on'\n if (coolingActive === 'on') {\n time = 'night'\n } else {\n time = 'day'\n }\n} else if (type === 'sleep' && payload === 'on') {\n setDisplay = 'turn_off'\n time = 'bedtime'\n} else {\n time = msg.time\n}\n\nif (topic === 'emmabedroom-wakeup') {\n setSleep = 'turn_off'\n}\n\n// Day Time\nif (time === 'day') {\n if (type === 'auto') {\n setCool = 'turn_off'\n }\n if (earlyNight === 'off') {\n if (ac === 'on') {\n if (danger === 'Extreme') {\n setTemp = nightTemp\n setEco = \"turn_on\"\n setHvac = \"cool\"\n } else if (hotDay === 'on' || heatWarning === 'on') {\n setTemp = dayTemp\n setEco = \"turn_on\"\n setHvac = \"cool\"\n } else {\n setTemp = nightTemp\n setEco = \"turn_on\"\n setHvac = \"off\"\n }\n }\n } else if (earlyNight === 'on') {\n if (ac === 'on') {\n if (danger === 'Extreme') {\n setTemp === bedTemp\n } else {\n setTemp = nightTemp\n }\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'fan') {\n setHvac = 'fan_only'\n } else {\n setHvac = 'off'\n }\n if (hotDay === 'on') {\n setEco = 'turn_off'\n } else {\n setEco = 'turn_on'\n }\n }\n }\n// Night Time\n} else if (time === 'night') {\n // If this is being run at scheduled time, turn on input_boolean.emma_bedroom_cooling_on\n if (type === 'auto') {\n setCool = 'turn_on'\n }\n // Decide temperature\n if (type === 'sleep' && payload === 'off') {\n setTemp = dayTemp\n } else if (danger === 'Extreme') {\n setTemp = bedTemp\n } else {\n setTemp = nightTemp\n }\n // Decide eco mode\n if (sleeping === 'on') {\n setEco = 'turn_off'\n } else {\n setEco = 'turn_on'\n }\n // Decide HVAC mode\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'Fan') {\n setHvac = 'fan_only'\n } else {\n setHvac = 'off'\n }\n// Bed Time\n} else if (time === 'bedtime') {\n setPeople = 'turn_off'\n if (ac === 'on') {\n setTemp = bedTemp\n setEco = 'turn_off'\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'Fan') {\n setHvac = \"fan_only\"\n } else if (schedMode === 'White Noise') {\n setWhiteNoise = 'turn_on'\n if (ac === 'on') {\n setHvac = 'off'\n }\n } else {\n setHvac = \"off\"\n }\n }\n}\nnode.log(\"Emma Bedroom Climate: Decision Logic Complete\")\n\n// Define message payloads\n\nlet sendCool = {\n \"payload\": {\n \"action\": `input_boolean.${setCool}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_bedroom_cooling_on\"]\n },\n \"data\": {}\n }\n}\n\nlet sendSleep = {\n \"payload\": {\n \"action\": `input_boolean.${setSleep}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_sleeping\"]\n },\n \"data\": {}\n }\n}\n\nlet sendPeople = {\n \"payload\": {\n \"action\": `input_boolean.${setPeople}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_awake\"]\n },\n \"data\": {}\n }\n}\n\nlet sendDisplay = {\n \"payload\": {\n \"action\": `switch.${setDisplay}`,\n \"target\": {\n \"entity_id\": [\"switch.emma_bedroom_aircon_display\"]\n },\n \"data\": {}\n }\n}\n\nlet sendWhiteNoise = {\n \"payload\": {\n \"action\": `input_boolean.${setWhiteNoise}`,\n \"target\": {\n \"entity_id\": whiteNoiseEntity\n },\n \"data\": {}\n }\n}\n\nlet notify = {\n \"topic\": topic,\n \"nighttemp\": nightTemp,\n \"schedMode\": schedMode\n}\n\nlet sendHvac = {\n \"payload\": {\n \"action\": \"climate.set_hvac_mode\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"hvac_mode\": setHvac\n }\n }\n}\n\nlet sendTemp = {\n \"payload\": {\n \"action\": \"climate.set_temperature\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"temperature\": setTemp\n }\n }\n}\n\nlet sendEco = {\n \"payload\": {\n \"action\": `switch.${setEco}`,\n \"target\": {\n \"entity_id\": [\"switch.emma_bedroom_aircon_eco_mode\"]\n },\n \"data\": {}\n }\n}\n\nlet sendAcFan = {\n \"payload\": {\n \"action\": \"climate.set_fan_mode\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"fan_mode\": \"auto\"\n }\n }\n}\n\nnode.log(\"Emma Bedroom Climate: Message Payloads Defined\")\n\n// Send some parameters to flow context for use in other nodes\n\nif (setHvac.length > 0) {\n flow.set('emmaBedroom.airconHvacMode',setHvac,'diskCon')\n}\n\nif (setEco.length > 0) {\n flow.set('emmaBedroom.airconPreset',setEco,'diskCon')\n}\n\nif (setTemp.length > 0) {\n flow.set('emmaBedroom.airconTargetTemp',setTemp,'diskCon')\n}\n\nif (setDisplay.length > 0) {\n flow.set('emmaBedroom.airconDisplay',setDisplay,'diskCon')\n}\n\nflow.set('emmaBedroom.airconFanMode','auto','diskCon')\n\n// Log the parameters that were chosen, for debugging purposes\n\nnode.log(\"----- Emma Bedroom Climate: Set Parameters -----\")\nnode.log(`setTemp: ${setTemp}`)\nnode.log(`setEco: ${setEco}`)\nnode.log(`setHvac: ${setHvac}`)\nnode.log(`setCool: ${setCool}`)\nnode.log(`setSleep: ${setSleep}`)\nnode.log(`setPeople: ${setPeople}`)\nnode.log(`setDisplay: ${setDisplay}`)\nnode.log(`time: ${time}`)\nnode.log(`type: ${type}`)\nnode.log(`topic: ${topic}`)\nnode.log(\"----- Emma Bedroom Climate: End Parameters -----\")\n\n// If this was an automated trigger, set the cooling context for the bedroom accordingly.\n\nif (type === 'auto' && time != 'bedtime') {\n node.send([null, sendCool, null])\n node.log(\"Emma Bedroom Climate: Cooling Context Set\")\n}\n\nif (type === 'auto' && allowed === 'on' && meltdown === 'off' && vacation === 'off') {\n node.log(\"Emma Bedroom Climate: Auto\")\n if (sleeping === 'on' && topic != 'emmabedroom-wakeup') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (sleep mode)\" })\n node.log(\"Emma Bedroom Climate: Blocked (sleep mode)\")\n } else {\n if (topic === 'emmabedroom-cooling' && ac === 'on') {\n node.status({ fill: \"green\", shape: \"dot\", text: \"Cooling Schedule\" })\n node.send([[sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Cooling\")\n } else if (topic === 'emmabedroom-bedtime') {\n node.send([null, sendPeople, null])\n node.status({ fill: \"green\", shape: \"dot\", text: \"Bedtime\" })\n node.log(\"Emma Bedroom Climate: Auto/Bedtime\")\n if (validSchedModesAC.includes(schedMode) && ac === 'on') {\n node.send([[sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Bedtime/AC\")\n } else if (schedMode === 'White Noise') {\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Auto/Bedtime/White Noise\")\n }\n } else if (topic === 'emmabedroom-wakeup') {\n node.status({ fill: \"green\", shape: \"dot\", text: \"Wakeup Schedule\" })\n node.log(\"Emma Bedroom Climate: Auto/Wakeup\")\n if (sleeping === 'off') {\n context.set(\"isWakeup\", false)\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/Sleep Off\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/AC On\")\n }\n } else if (sleeping === 'on') {\n context.set(\"isWakeup\", true)\n node.send([null, sendSleep, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/Sleep On\")\n }\n }\n }\n// Manual Responses\n} else if (type === 'manual') {\n node.log(\"Emma Bedroom Climate: Manual\")\n if (time === 'night') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Night\" })\n node.log(\"Emma Bedroom Climate: Manual/Night\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Night/AC\")\n }\n } else if (time === 'day') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Day\" })\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Manual/Day\")\n if (ac === 'on') {\n node.send([[sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Day/AC\")\n }\n } else if (time === 'bedtime') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Bedtime\" })\n node.log(\"Emma Bedroom Climate: Manual/Bedtime\")\n if (validSchedModesAC.includes(schedMode) && ac === 'on') {\n node.send([[sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Bedtime/AC\")\n } else if (schedMode === 'White Noise') {\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Manual/Bedtime/White Noise\")\n }\n }\n context.set(\"isWakeup\", false)\n// Sleep Switch Responses\n} else if (type === 'sleep') {\n if (payload === 'off') {\n node.send([null, [sendPeople, sendWhiteNoise], null])\n }\n node.log(\"Emma Bedroom Climate: Sleep\")\n if (time === 'bedtime') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Sleep\" })\n node.send([null, [sendPeople, sendWhiteNoise], null])\n node.log(\"Emma Bedroom Climate: Sleep/Bedtime\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Sleep/Bedtime/AC\")\n }\n } else {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Wakeup\" })\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Sleep/Day\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(`Emma Bedroom Climate: Sleep/${convertToTitleCase(time)}/AC`)\n }\n }\n context.set(\"isWakeup\", false)\n} else if (meltdown === 'on') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Meltdown Protocol)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Meltdown Protocol)\")\n} else if (vacation === 'on') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Vacation Mode)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Vacation Mode)\")\n} else {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Automation Disabled)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Automation Disabled)\")\n}\n\nnode.log(\"Emma Bedroom Climate: Processing Complete\")\n", + "func": "node.log(\"Emma Bedroom Climate: Processing Started\");\n// pull in the necessary information\n\nconst states = global.get('homeassistant.homeAssistant.states')\nconst allowed = states['input_boolean.emma_bedroom_climate_protocol'].state\nconst ac = global.get('emmaBedroom.aircon.installed', \"diskCon\")\nconst temp = global.get(\"outdoorTemp.tempStr\")\nconst payload = msg.payload\nconst vacation = states['input_boolean.vacation_mode'].state\nconst dayTemp = states['input_number.emma_bedroom_daytime_temp'].state\nconst nightTemp = states['input_number.emma_bedroom_night_temp'].state\nconst bedTemp = states['input_number.emma_bedroom_bedtime_temp'].state\nconst schedMode = states['input_select.scheduled_climate_mode_emma_bedroom'].state\nconst sleeping = states['input_boolean.emma_sleeping'].state\nconst hotDay = states['input_boolean.hot_day'].state\nconst heatWarning = states[\"binary_sensor.heat_warning\"].state\nconst earlyNight = states[\"binary_sensor.early_night_mode\"].state\nconst danger = states['binary_sensor.heat_warning'].attributes.danger\nconst meltdown = states['input_boolean.meltdown_protocol'].state\nconst coolingActive = states['input_boolean.emma_bedroom_cooling_on'].state\nnode.log(\"Emma Bedroom Climate: Constants Set\")\n\n// Define reusable constants\n\nconst airconEntity = [\"climate.emma_bedroom_aircon\"]\nconst whiteNoiseEntity = [\"input_boolean.white_noise_emma_bedroom\"]\nconst validSchedModesAC = ['AC', 'Fan']\nconst peopleIDs = [\"input_boolean.emma_awake\"]\n\n// Helper function to convert a string to title case\n\nfunction convertToTitleCase(str) {\n if (!str) {\n return \"\";\n }\n return str.toLowerCase().replace(/\\b\\w/g, (s) => s.toUpperCase());\n}\n\n// init variables\n\nlet setTemp = dayTemp\nlet setAcFan = 'auto'\nlet setEco = 'turn_on'\nlet setHvac = 'cool'\nlet setCool = []\nlet setSleep = []\nlet setPeople = []\nlet setDisplay = 'turn_on'\nlet setWhiteNoise = \"turn_off\"\nlet time = []\nlet type = msg.type\nlet topic = msg.topic\nlet isWakeup = context.get(\"isWakeup\")\nnode.log(\"Emma Bedroom Climate: Variables Defined\")\n\n// Sleep Switch Handling\nif (type === 'sleep' && payload === 'off') {\n setDisplay = 'turn_on'\n setPeople = 'turn_on'\n if (coolingActive === 'on') {\n time = 'night'\n } else {\n time = 'day'\n }\n} else if (type === 'sleep' && payload === 'on') {\n setDisplay = 'turn_off'\n time = 'bedtime'\n} else {\n time = msg.time\n}\n\nif (topic === 'emmabedroom-wakeup') {\n setSleep = 'turn_off'\n}\n\n// Day Time\nif (time === 'day') {\n if (type === 'auto') {\n setCool = 'turn_off'\n }\n if (earlyNight === 'off') {\n if (ac === 'on') {\n if (danger === 'Extreme') {\n setTemp = nightTemp\n setEco = \"turn_on\"\n setHvac = \"cool\"\n } else if (hotDay === 'on' || heatWarning === 'on') {\n setTemp = dayTemp\n setEco = \"turn_on\"\n setHvac = \"cool\"\n } else {\n setTemp = nightTemp\n setEco = \"turn_on\"\n setHvac = \"off\"\n }\n }\n } else if (earlyNight === 'on') {\n if (ac === 'on') {\n if (danger === 'Extreme') {\n setTemp === bedTemp\n } else {\n setTemp = nightTemp\n }\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'fan') {\n setHvac = 'fan_only'\n } else {\n setHvac = 'off'\n }\n if (hotDay === 'on') {\n setEco = 'turn_off'\n } else {\n setEco = 'turn_on'\n }\n }\n }\n// Night Time\n} else if (time === 'night') {\n // If this is being run at scheduled time, turn on input_boolean.emma_bedroom_cooling_on\n if (type === 'auto') {\n setCool = 'turn_on'\n }\n // Decide temperature\n if (type === 'sleep' && payload === 'off') {\n setTemp = dayTemp\n } else if (danger === 'Extreme') {\n setTemp = bedTemp\n } else {\n setTemp = nightTemp\n }\n // Decide eco mode\n if (sleeping === 'on') {\n setEco = 'turn_off'\n } else {\n setEco = 'turn_on'\n }\n // Decide HVAC mode\n if (schedMode === 'AC') {\n setHvac = 'cool'\n } else if (schedMode === 'Fan') {\n setHvac = 'fan_only'\n } else {\n setHvac = 'off'\n }\n// Bed Time\n} else if (time === 'bedtime') {\n setPeople = 'turn_off'\n if (ac === 'on') {\n setTemp = bedTemp\n setEco = 'turn_off'\n if (schedMode === 'AC') {\n setHvac = 'cool'\n setDisplay = 'turn_off'\n } else if (schedMode === 'Fan') {\n setHvac = \"fan_only\"\n setDisplay = 'turn_off'\n } else if (schedMode === 'White Noise') {\n setWhiteNoise = 'turn_on'\n if (ac === 'on') {\n setHvac = 'off'\n }\n } else {\n setHvac = \"off\"\n }\n }\n}\nnode.log(\"Emma Bedroom Climate: Decision Logic Complete\")\n\n// Define message payloads\n\nlet sendCool = {\n \"payload\": {\n \"action\": `input_boolean.${setCool}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_bedroom_cooling_on\"]\n },\n \"data\": {}\n }\n}\n\nlet sendSleep = {\n \"payload\": {\n \"action\": `input_boolean.${setSleep}`,\n \"target\": {\n \"entity_id\": [\"input_boolean.emma_sleeping\"]\n },\n \"data\": {}\n }\n}\n\nlet sendPeople = {\n \"payload\": {\n \"action\": `input_boolean.${setPeople}`,\n \"target\": {\n \"entity_id\": peopleIDs\n },\n \"data\": {}\n }\n}\n\nlet sendDisplay = {\n \"payload\": {\n \"action\": `switch.${setDisplay}`,\n \"target\": {\n \"entity_id\": [\"switch.emma_bedroom_aircon_display\"]\n },\n \"data\": {}\n }\n}\n\nlet sendWhiteNoise = {\n \"payload\": {\n \"action\": `input_boolean.${setWhiteNoise}`,\n \"target\": {\n \"entity_id\": whiteNoiseEntity\n },\n \"data\": {}\n }\n}\n\nlet notify = {\n \"topic\": topic,\n \"nighttemp\": nightTemp,\n \"schedMode\": schedMode\n}\n\nlet sendHvac = {\n \"payload\": {\n \"action\": \"climate.set_hvac_mode\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"hvac_mode\": setHvac\n }\n }\n}\n\nlet sendTemp = {\n \"payload\": {\n \"action\": \"climate.set_temperature\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"temperature\": setTemp\n }\n }\n}\n\nlet sendEco = {\n \"payload\": {\n \"action\": `switch.${setEco}`,\n \"target\": {\n \"entity_id\": [\"switch.emma_bedroom_aircon_eco_mode\"]\n },\n \"data\": {}\n }\n}\n\nlet sendAcFan = {\n \"payload\": {\n \"action\": \"climate.set_fan_mode\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"fan_mode\": setAcFan\n }\n }\n}\n\nnode.log(\"Emma Bedroom Climate: Message Payloads Defined\")\n\n// Send some parameters to flow context for use in other nodes\n\nif (setHvac.length > 0) {\n flow.set('emmaBedroom.airconHvacMode',setHvac,'diskCon')\n}\n\nif (setEco.length > 0) {\n flow.set('emmaBedroom.airconEco',setEco,'diskCon')\n}\n\nif (setTemp.length > 0) {\n flow.set('emmaBedroom.airconTargetTemp',setTemp,'diskCon')\n}\n\nif (setDisplay.length > 0) {\n flow.set('emmaBedroom.airconDisplay',setDisplay,'diskCon')\n}\n\nflow.set('emmaBedroom.airconFanMode',setAcFan,'diskCon')\n\n// Log the parameters that were chosen, for debugging purposes\n\nnode.log(\"----- Emma Bedroom Climate: Set Parameters -----\")\nnode.log(`setTemp: ${setTemp}`)\nnode.log(`setAcFan: ${setAcFan}`)\nnode.log(`setEco: ${setEco}`)\nnode.log(`setHvac: ${setHvac}`)\nnode.log(`setCool: ${setCool}`)\nnode.log(`setSleep: ${setSleep}`)\nnode.log(`setPeople: ${setPeople}`)\nnode.log(`setDisplay: ${setDisplay}`)\nnode.log(`time: ${time}`)\nnode.log(`type: ${type}`)\nnode.log(`topic: ${topic}`)\nnode.log(\"----- Emma Bedroom Climate: End Parameters -----\")\n\n// If this was an automated trigger, set the cooling context for the bedroom accordingly.\n\nif (type === 'auto' && time != 'bedtime') {\n node.send([null, sendCool, null])\n node.log(\"Emma Bedroom Climate: Cooling Context Set\")\n}\n\nif (type === 'auto' && allowed === 'on' && meltdown === 'off' && vacation === 'off') {\n node.log(\"Emma Bedroom Climate: Auto\")\n if (sleeping === 'on' && topic != 'emmabedroom-wakeup') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (sleep mode)\" })\n node.log(\"Emma Bedroom Climate: Blocked (sleep mode)\")\n } else {\n if (topic === 'emmabedroom-cooling' && ac === 'on') {\n node.status({ fill: \"green\", shape: \"dot\", text: \"Cooling Schedule\" })\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Cooling\")\n } else if (topic === 'emmabedroom-bedtime') {\n node.send([null, sendPeople, null])\n node.status({ fill: \"green\", shape: \"dot\", text: \"Bedtime\" })\n node.log(\"Emma Bedroom Climate: Auto/Bedtime\")\n if (validSchedModesAC.includes(schedMode) && ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Bedtime/AC\")\n } else if (schedMode === 'White Noise') {\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Auto/Bedtime/White Noise\")\n }\n } else if (topic === 'emmabedroom-wakeup') {\n node.status({ fill: \"green\", shape: \"dot\", text: \"Wakeup Schedule\" })\n node.log(\"Emma Bedroom Climate: Auto/Wakeup\")\n if (sleeping === 'off') {\n context.set(\"isWakeup\", false)\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/Sleep Off\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/AC On\")\n }\n } else if (sleeping === 'on') {\n context.set(\"isWakeup\", true)\n node.send([null, sendSleep, null])\n node.log(\"Emma Bedroom Climate: Auto/Wakeup/Sleep On\")\n }\n }\n }\n// Manual Responses\n} else if (type === 'manual') {\n node.log(\"Emma Bedroom Climate: Manual\")\n if (time === 'night') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Night\" })\n node.log(\"Emma Bedroom Climate: Manual/Night\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Night/AC\")\n }\n } else if (time === 'day') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Day\" })\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Manual/Day\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Day/AC\")\n }\n } else if (time === 'bedtime') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Manual Bedtime\" })\n node.log(\"Emma Bedroom Climate: Manual/Bedtime\")\n if (validSchedModesAC.includes(schedMode) && ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Manual/Bedtime/AC\")\n } else if (schedMode === 'White Noise') {\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Manual/Bedtime/White Noise\")\n }\n }\n context.set(\"isWakeup\", false)\n// Sleep Switch Responses\n} else if (type === 'sleep') {\n if (payload === 'off') {\n node.send([null, [sendPeople, sendWhiteNoise], null])\n }\n node.log(\"Emma Bedroom Climate: Sleep\")\n if (time === 'bedtime') {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Sleep\" })\n node.send([null, [sendPeople, sendWhiteNoise], null])\n node.log(\"Emma Bedroom Climate: Sleep/Bedtime\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(\"Emma Bedroom Climate: Sleep/Bedtime/AC\")\n }\n } else {\n node.status({ fill: \"blue\", shape: \"dot\", text: \"Wakeup\" })\n node.send([null, sendWhiteNoise, null])\n node.log(\"Emma Bedroom Climate: Sleep/Day\")\n if (ac === 'on') {\n node.send([[sendDisplay, sendHvac, sendTemp, sendAcFan, sendEco], null, null])\n node.log(`Emma Bedroom Climate: Sleep/${convertToTitleCase(time)}/AC`)\n }\n }\n context.set(\"isWakeup\", false)\n} else if (meltdown === 'on') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Meltdown Protocol)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Meltdown Protocol)\")\n} else if (vacation === 'on') {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Vacation Mode)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Vacation Mode)\")\n} else {\n node.status({ fill: \"red\", shape: \"ring\", text: \"Blocked (Automation Disabled)\" })\n node.log(\"Emma Bedroom Climate: Blocked (Automation Disabled)\")\n}\n\nnode.log(\"Emma Bedroom Climate: Processing Complete\")\n", "outputs": 3, "timeout": "", "noerr": 0, @@ -24620,7 +24623,7 @@ "z": "72f99805df043603", "g": "5802ea32c1de36e8", "name": "AC Settings Debug", - "active": false, + "active": true, "tosidebar": true, "console": false, "tostatus": false, @@ -24670,7 +24673,7 @@ "y": 1840, "wires": [ [ - "71b1c6e934b85637" + "f1f2ac113fe2b639" ] ] }, @@ -24680,18 +24683,19 @@ "z": "72f99805df043603", "g": "5802ea32c1de36e8", "name": "Watchdog", - "func": "const airconState = msg.payload.state\nconst airconAttributes = msg.payload.attributes\nconst airconPreset = airconAttributes['preset_mode']\nconst airconEco = airconAttributes['eco_mode']\nconst airconTargetTemp = airconAttributes['temperature']\nconst airconFanMode = airconAttributes['fan_mode']\nconst airconDisplay = airconAttributes['screen_display']\n\nmsg.payload = {\n \"airconState\": airconState,\n \"airconPreset\": airconPreset,\n \"airconEco\": airconEco,\n \"airconTargetTemp\": airconTargetTemp,\n \"airconFanMode\": airconFanMode,\n \"airconDisplay\": airconDisplay\n}\n\nreturn msg", + "func": "const states = global.get('homeassistant.homeAssistant.states')\nconst sleeping = states['input_boolean.emma_sleeping'].state\n\nif (sleeping !== 'on') {\n node.status({fill:'red',shape:'ring',text:'Emma not sleeping, watchdog disabled'})\n return null\n}\n\nconst airconEntity = ['climate.emma_bedroom_aircon']\n\n// Gather relevant attributes from the aircon entity\n\nconst airconState = msg.payload.state\nconst airconAttributes = msg.payload.attributes\nconst airconEco = airconAttributes['eco_mode']\nconst airconTargetTemp = airconAttributes['temperature']\nconst airconFanMode = airconAttributes['fan_mode']\nconst airconDisplay = airconAttributes['screen_display']\n\n// Gather the last explicitly sent configuration from the context\n\nconst airconHvacCfg = flow.get('emmaBedroom.airconHvacMode', 'diskCon')\nconst airconEcoCfg = flow.get('emmaBedroom.airconEco', 'diskCon')\nconst airconTargetTempCfg = parseFloat(flow.get('emmaBedroom.airconTargetTemp', 'diskCon'))\nconst airconFanModeCfg = flow.get('emmaBedroom.airconFanMode', 'diskCon')\nconst airconDisplayCfg = flow.get('emmaBedroom.airconDisplay', 'diskCon')\n\n// Compare the current state with the last sent configuration\n\nlet ecoMatch = {}\nlet displayMatch = {}\nlet counter = 0\n\nlet sendDisplay = {\n \"payload\": {\n \"action\": `switch.${airconDisplayCfg}`,\n \"target\": {\n \"entity_id\": [\"switch.emma_bedroom_aircon_display\"]\n },\n \"data\": {}\n }\n}\n\nlet sendHvac = {\n \"payload\": {\n \"action\": \"climate.set_hvac_mode\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"hvac_mode\": airconHvacCfg\n }\n }\n}\n\nlet sendTemp = {\n \"payload\": {\n \"action\": \"climate.set_temperature\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"temperature\": airconTargetTempCfg\n }\n }\n}\n\nlet sendEco = {\n \"payload\": {\n \"action\": `switch.${airconEcoCfg}`,\n \"target\": {\n \"entity_id\": [\"switch.emma_bedroom_aircon_eco_mode\"]\n },\n \"data\": {}\n }\n}\n\nlet sendAcFan = {\n \"payload\": {\n \"action\": \"climate.set_fan_mode\",\n \"target\": {\n \"entity_id\": airconEntity\n },\n \"data\": {\n \"fan_mode\": airconFanModeCfg\n }\n }\n}\n\n// Fix hvac mode if mismatch\n\nif (airconState != airconHvacCfg) {\n node.log(`Emma Bedroom Aircon HVAC mode mismatch: ${airconState} vs ${airconHvacCfg}`)\n node.send(sendHvac)\n counter += 1\n}\n\n// Fix eco mode if mismatch\n\nif (airconEco === false && airconEcoCfg === 'turn_off') {\n ecoMatch = true\n} else if (airconEco === true && airconEcoCfg === 'turn_on') {\n ecoMatch = true\n} else {\n ecoMatch = false\n}\n\nif (ecoMatch === false) {\n node.log(`Emma Bedroom Aircon Eco mode mismatch: ${airconEco} vs ${airconEcoCfg}`)\n node.send(sendEco)\n counter += 1\n}\n\n// Fix target temperature if mismatch\n\nif (airconTargetTemp != airconTargetTempCfg) {\n node.log(`Target temperature mismatch: ${airconTargetTemp} vs ${airconTargetTempCfg}`)\n node.send(sendTemp)\n counter += 1\n}\n\n// Fix fan mode if mismatch\n\nif (airconFanMode != airconFanModeCfg) {\n node.log(`Fan mode mismatch: ${airconFanMode} vs ${airconFanModeCfg}`)\n node.send(sendAcFan)\n counter += 1\n}\n\n// Fix display mode if mismatch\n\nif (airconDisplay === false && airconDisplayCfg === 'turn_off') {\n displayMatch = true\n} else if (airconDisplay === true && airconDisplayCfg === 'turn_on') {\n displayMatch = true\n} else {\n displayMatch = false\n}\n\nif (displayMatch === false) {\n node.log(`Emma Bedroom Aircon Display mode mismatch: ${airconDisplay} vs ${airconDisplayCfg}`)\n node.send(sendDisplay)\n counter += 1\n}\n\nif (counter > 0) {\n node.log(`Emma Bedroom Aircon: ${counter} configuration(s) fixed`)\n node.status({fill:'yellow',shape:'dot',text:`${counter} config(s) fixed`})\n} else {\n node.status({fill:'green',shape:'dot',text:`No config changes`})\n}", "outputs": 1, "timeout": 0, "noerr": 0, "initialize": "", "finalize": "", "libs": [], - "x": 480, + "x": 460, "y": 1840, "wires": [ [ - "334a79e6f5f3a16a" + "334a79e6f5f3a16a", + "b96ad84b569ac360" ] ] }, @@ -24701,7 +24705,7 @@ "z": "72f99805df043603", "g": "5802ea32c1de36e8", "name": "Watchdog Debug", - "active": false, + "active": true, "tosidebar": true, "console": false, "tostatus": false, @@ -24922,6 +24926,64 @@ "y": 660, "wires": [] }, + { + "id": "f1f2ac113fe2b639", + "type": "delay", + "z": "72f99805df043603", + "g": "5802ea32c1de36e8", + "name": "", + "pauseType": "rate", + "timeout": "5", + "timeoutUnits": "seconds", + "rate": "1", + "nbRateUnits": "1", + "rateUnits": "second", + "randomFirst": "1", + "randomLast": "5", + "randomUnits": "seconds", + "drop": true, + "allowrate": false, + "outputs": 1, + "x": 325, + "y": 1840, + "wires": [ + [ + "71b1c6e934b85637" + ] + ], + "l": false + }, + { + "id": "b96ad84b569ac360", + "type": "link out", + "z": "72f99805df043603", + "g": "5802ea32c1de36e8", + "name": "Emma Climate Watchdog", + "mode": "link", + "links": [ + "894e510fd44b01e0" + ], + "x": 595, + "y": 1780, + "wires": [] + }, + { + "id": "894e510fd44b01e0", + "type": "link in", + "z": "72f99805df043603", + "g": "5802ea32c1de36e8", + "name": "Emma Bedroom Aircon Settings - In", + "links": [ + "b96ad84b569ac360" + ], + "x": 475, + "y": 1480, + "wires": [ + [ + "53b440a31d30df1b" + ] + ] + }, { "id": "48b8af9419ed18d6", "type": "subflow:384eaf2f63dcb5db",