/* groovylint-disable NglParseError, ImplicitReturnStatement, InsecureRandom, MethodReturnTypeRequired, MethodSize, ParameterName, PublicMethodsBeforeNonPublicMethods, StaticMethodsBeforeInstanceMethods, UnnecessaryGroovyImport, UnnecessaryObjectReferences, UnusedImport, VariableName *//** * Tuya Zigbee Chlorine Meter- driver for Hubitat Elevation * * https://community.hubitat.com/t/dynamic-capabilities-commands-and-attributes-for-drivers/98342 * * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except * in compliance with the License. You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License * for the specific language governing permissions and limitations under the License. * * ver. 3.3.0 2024-08-03 kkossev - first test version * ver. 3.3.1 2024-08-31 kkossev - added tuyaDataQuery; added dp 103 104 114 115 116 118 decoding; invalid freeChlorine value -1.0 (0xFFFFFFFFF) returned as 0 (zero), added automatic polling (configurable) * ver. 3.3.2 2024-09-06 kkossev - debug is off by default; freeChlorine is divided by 10; * ver. 3.4.0 2025-05-24 kkossev - HE platfrom version 2.4.1.x decimal preferences patch/workaround.* * TODO: */ static String version() { "3.4.0" } static String timeStamp() { "2025/05/24 6:46 PM" } @Field static final Boolean _DEBUG = false @Field static final Boolean _TRACE_ALL = false // trace all messages, including the spammy ones @Field static final Boolean DEFAULT_DEBUG_LOGGING = false // disable it for production deviceType = "MultiMeter" @Field static final String DEVICE_TYPE = "MultiMeter" metadata { definition ( name: 'Tuya Zigbee Chlorine Meter', importUrl: 'https://raw.githubusercontent.com/kkossev/Hubitat/development/Drivers/Tuya%20Zigbee%20Chlorine%20Meter/Tuya_Zigbee_Chlorine_Meter_lib_included.groovy', namespace: 'kkossev', author: 'Krassimir Kossev', singleThreaded: true ) { // no standard capabilities attribute 'tds', 'number' // Total Dissolved Solids attribute 'temperature', 'number' // Temperature attribute 'battery', 'number' // Battery level remaining attribute 'ph', 'number' // pH value attribute 'ec', 'number' // Electrical conductivity attribute 'orp', 'number' // Oxidation Reduction Potential value attribute 'freeChlorine', 'number' // Free chlorine value attribute 'backlightvalue', 'number' // Bbacklight value attribute 'phMmax', 'number' // pH maximal value attribute 'phMmin', 'number' // pH minimal value attribute 'ecMmax', 'number' // Electrical Conductivity maximal value attribute 'ecMmin', 'number' // Electrical Conductivity minimal value attribute 'orpMmax', 'number' // Oxidation Reduction Potential maximal value attribute 'orpMmin', 'number' // Oxidation Reduction Potential minimal value attribute 'freeChlorineMax', 'number' // Free Chlorine maximal value attribute 'freeChlorineMin', 'number' // Free Chlorine minimal value attribute 'salinity', 'number' // Salt value attribute 'backlightvalue', 'number' // Bbacklight value // no commands if (_DEBUG) { command 'tuyaDataQuery' } // itterate through all the figerprints and add them on the fly deviceProfilesV3.each { profileName, profileMap -> if (profileMap.fingerprints != null) { if (profileMap.device?.isDepricated != true) { profileMap.fingerprints.each { fingerprint it } } } } } preferences { if (device) { // input(name: 'info', type: 'hidden', title: "For more info, click on this link to visit the WiKi page") } input name: 'txtEnable', type: 'bool', title: 'Enable descriptionText logging', defaultValue: true, description: 'Enables events logging.' input name: 'logEnable', type: 'bool', title: 'Enable debug logging', defaultValue: DEFAULT_DEBUG_LOGGING, description: 'Turns on debug logging for 24 hours.' // the rest of the preferences are inputIt from the deviceProfileLib and from the included libraries if (device) { input name: 'pollingInterval', type: 'enum', title: 'Polling Interval', options: PollingIntervalOpts.options, defaultValue: PollingIntervalOpts.defaultValue, required: true, description: 'Changes how often the hub will poll the sensor.' } } } @Field static String ttStyleStr = '' @Field static final Map PollingIntervalOpts = [ defaultValue: 300, options : [0: 'Disabled', 5: 'Every 5 seconds (DONT DO THAT!)', 60: 'Every minute (not recommended)', 120: 'Every 2 minutes', 300: 'Every 5 minutes (default)', 600: 'Every 10 minutes', 900: 'Every 15 minutes', 1800: 'Every 30 minutes', 3600: 'Every 1 hour'] ] /* Measures : PH : test ranges: 0.0-14.0ph; Resolution: 0.1ph; Accuracy: ±0.1ph CL : test ranges: 0.0-4.0mg/L; Resolution: 0.1mg/L; Accuracy: ±0.1mg/L Salt : test ranges: 0-999ppm, 1000-9990ppm; Resolution: 1ppm, 10ppm; Accuracy: ±2% F.S EC : test ranges: 0-2000us/c m , 2000-9990uS/c m,10.01- 19.99mS/c m; Resolution: 1uS/c m 10uS/c m 0.01mS/C M; Accuracy: ±2% F.S. Total Dissolved Solids : test ranges: 0-999ppm,1000- 9990pp, Resolution: 1ppm 10ppm; Accuracy: ±2% F.S. ORP : test ranges: -999mv ~+999mv; Resolution: 1mv; Accuracy:15mv Temperature : test ranges: 0.0℃-50.0℃ 32.0℉-122.0℉; Resolution: 0.1℃/0.1℉; Accuracy: ±0.5℃ */ @Field static final Map deviceProfilesV3 = [ // https://github.com/Koenkk/zigbee2mqtt/issues/18704 // https://community.home-assistant.io/t/pool-monitoring-device-yieryi-ble-yl01-zigbee-ph-orp-free-chlorine-salinity-etc/659545/10 // https://github.com/Koenkk/zigbee2mqtt/issues/18704#issuecomment-1732263086 // https://github.com/zigbeefordomoticz/z4d-certified-devices/blob/e65463300dda776145ca4b2953ebe162c2f60b3d/z4d_certified_devices/Certified/Tuya/TS0601-BLE-YL01.json#L7 'CHLORINE_METER_BLE_YL01' : [ description : 'BLE_YL01 Tuya Zigbee Chlorine Meter', models : ['TS0601'], device : [type: 'Sensor', powerSource: 'dc', isSleepy:false], // check powerSource capabilities : ['Battery': true, 'TemperatureMeasurement': true], preferences : ['phMmax': '106', 'phMmin': '107', 'ecMmax': '108', 'ecMmin': '109', 'orpMmax': '110', 'orpMmin': '111', 'freeChlorineMax': '112', 'freeChlorineMin': '113'], // "Param": { "tempCompensation": 0, "ph7Compensation": 0, "ecCompensation": 0, "orpCompensation": 0 } commands : ['resetStats':'resetStats', 'refresh':'refresh', 'initialize':'initialize', 'updateAllPreferences': 'updateAllPreferences', 'resetPreferencesToDefaults':'resetPreferencesToDefaults', 'validateAndFixPreferences':'validateAndFixPreferences', 'printFingerprints':'printFingerprints', 'printPreferences':'printPreferences'], fingerprints : [ [profileId:'0104', endpointId:'01', inClusters:'0000,0004,0005,EF00', outClusters:'0019,000A', model:'TS0601', manufacturer:'_TZE200_v1jqz5cy', deviceJoinName: 'BLE_YL01 Tuya Zigbee Chlorine Meter'], [profileId:'0104', endpointId:'01', inClusters:'0000,0004,0005,EF00', outClusters:'0019,000A', model:'TS0601', manufacturer:'_TZE200_d9mzkhoq', deviceJoinName: 'BLE_YL01 Tuya Zigbee Chlorine Meter'], ], tuyaDPs: [ [dp:1, name:'tds', type:'number', rw: 'ro', scale:1, unit:'ppm', description:'Total Dissolved Solids'], [dp:2, name:'temperature', type:'decimal', rw: 'ro', scale:10, unit:'C', description:'Temperature'], [dp:7, name:'battery', type:'number', rw: 'ro', scale:1, unit:'%', description:'Battery level remaining'], [dp:10, name:'ph', type:'decimal', rw: 'ro', scale:100, unit:'pH', description:'pH value'], // 'pH value, if the pH value is lower than 6.5, it means that the water quality is too acidic and has impurities, and it is necessary to add disinfectant water for disinfection [dp:11, name:'ec', type:'decimal', rw: 'ro', scale:1, unit:'µS/cm', description:'Electrical conductivity'], [dp:101, name:'orp', type:'decimal', rw: 'ro', scale:1, unit:'mV', description:'Oxidation Reduction Potential value'], // 'Oxidation Reduction Potential value. If the ORP value is above 850mv, it means that the disinfectant has been added too much, and it is necessary to add water or change the water for neutralization. If the ORP value is below 487mv, it means that too little disinfectant has been added and the pool needs to be disinfected again' [dp:102, name:'freeChlorine', preProc:'checkInvalidValue', type:'decimal', rw: 'ro', scale:10, unit:'mg/L', description:'Free chlorine value'], // The water in the swimming pool should be between 6.5-8ph and ORP should be between 487-840mv, and the chlorine value will be displayed normally. Chlorine will not be displayed if either value is out of range [dp:103, name:'phCalibration1', type:'number', rw: 'ro', scale:1, unit:'', description:'pH Calibration 1'], // "67": { "sensor_type": "phCalibration1" }, [dp:104, name:'backlightStatus', type:'number', rw: 'ro', scale:1, unit:'', description:'Backlight status'], // "68": { "store_tuya_attribute": "backlight_status", "EvalExp": "(value)" }, [dp:105, name:'backlightLevel', type:'number', rw: 'ro', scale:1, unit:'', description:'Backlight level'], // "69": { "store_tuya_attribute": "backlight_level", "EvalExp": "(value)" }, dp:105 [dp:106, name:'phMmax', type:'decimal', rw: 'rw', min:0, max:20, /*defVal:14.0,*/ scale:10, unit:'pH', title:'pH maximal value'], [dp:107, name:'phMmin', type:'decimal', rw: 'rw', min:0, max:20, /*defVal:0.0,*/ scale:10, unit:'pH', title:'pH minimal value'], [dp:108, name:'ecMmax', type:'decimal', rw: 'rw', min:0, max:20000, /*defVal:20000.0,*/ scale:1, unit:'µS/cm', title:'Electrical Conductivity maximal value'], [dp:109, name:'ecMmin', type:'decimal', rw: 'rw', min:0, max:100, /*defVal:0.0,*/ scale:1, unit:'µS/cm', title:'Electrical Conductivity minimal value'], [dp:110, name:'orpMmax', type:'decimal', rw: 'rw', min:0, max:1000, /*defVal:999.0,*/ scale:1, unit:'mV', title:'Oxidation Reduction Potential maximal value'], [dp:111, name:'orpMmin', type:'decimal', rw: 'rw', min:0, max:1000, /*defVal:0.0,*/ scale:1, unit:'mV', title:'Oxidation Reduction Potential minimal value'], [dp:112, name:'freeChlorineMax', type:'decimal', rw: 'rw', min:0, max:15, /*defVal:20.0,*/ scale:10, unit:'mg/L', title:'Free Chlorine maximal value'], [dp:113, name:'freeChlorineMin', type:'decimal', rw: 'rw', min:0, max:15, /*defVal:20.0,*/ scale:10, unit:'mg/L', title:'Free Chlorine minimal value'], [dp:114, name:'phCalibration2', type:'number', rw: 'ro', scale:1, unit:'', description:'pH Calibration 2'], // "72": { "sensor_type": "phCalibration2" }, [dp:115, name:'ecCalibration', type:'number', rw: 'ro', scale:1, unit:'', description:'EC Calibration'], // "73": { "sensor_type": "ecCalibration" }, [dp:116, name:'orpCalibration', type:'number', rw: 'ro', scale:1, unit:'', description:'ORP Calibration'], // "74": { "sensor_type": "orpCalibration" }, [dp:117, name:'salinity', type:'decimal', rw: 'ro', scale:1, unit:'gg', description:'Salt value'], [dp:118, name:'salinityCalibration', type:'number', rw: 'ro', scale:1, unit:'', description:'Salinity? Calibration ?(0x76)'], // "76": { "sensor_type": "salinityCalibration" }, ], refresh: ['refreshQueryAllTuyaDP'], configuration : ['battery': false], deviceJoinName: 'BLE_YL01 Tuya Zigbee Chlorine Meter' ] // second manufacturer ? // https://www.amazon.com/YINMIK-Chlorine-Swimming-Salinity-Inground/dp/B0C2T8YLYW // https://github.com/Koenkk/zigbee-herdsman-converters/pull/7613 ] Number checkInvalidValue(Number value) { if (value < 0) { logDebug "freeChlorine Invalid value -1.0 detected, returning zero!" return 0 } return value } // called from standardProcessTuyaDP in the commonLib for each Tuya dp report in a Zigbee message // should always return true, as we are processing all the dp reports here boolean customProcessTuyaDp(final Map descMap, final int dp, final int dp_id, final int fncmd, final int dp_len=0) { logDebug "customProcessTuyaDp: dp=${dp} dp_id=${dp_id} fncmd=${fncmd} dp_len=${dp_len} descMap.data = ${descMap?.data}" if (processTuyaDPfromDeviceProfile(descMap, dp, dp_id, fncmd, dp_len) == true) { return true // sucessfuly processed from the deviceProfile } logWarn "NOT PROCESSED from deviceProfile Tuya cmd: dp=${dp} value=${fncmd} descMap.data = ${descMap?.data}" localProcessTuyaDP(descMap, dp, dp_id, fncmd, dp_len) return true } void localProcessTuyaDP(final Map descMap, final int dp, final int dp_id, final int fncmd, final int dp_len) { switch (dp) { default : logDebug "NOT PROCESSED Tuya cmd: dp=${dp} value=${fncmd} descMap.data = ${descMap?.data}" break } } // called from processFoundItem in the deviceProfileLib void customProcessDeviceProfileEvent(final Map descMap, final String name, valueScaled, final String unitText, final String descText) { logTrace "customProcessDeviceProfileEvent(${name}, ${valueScaled}) called" Map eventMap = [name: name, value: valueScaled, unit: unitText, descriptionText: descText, type: 'physical', isStateChange: true] switch (name) { default : sendEvent(name : name, value : valueScaled, unit:unitText, descriptionText: descText, type: 'physical', isStateChange: true) // attribute value is changed - send an event ! logTrace "event ${name} sent w/ value ${valueScaled}" logInfo "${descText}" // TODO - send info log only if the value has changed? // TODO - check whether Info log will be sent also for spammy clusterAttribute ? break } } List customRefresh() { logDebug "customRefresh()" List cmds = [] List devProfCmds = refreshFromDeviceProfileList() if (devProfCmds != null && !devProfCmds.isEmpty()) { cmds += devProfCmds } return cmds } void customUpdated() { logDebug "customUpdated()" List cmds = [] if (settings?.forcedProfile != null) { if (this.respondsTo('getProfileKey') == false) { logWarn "getProfileKey() is not defined in the driver" } else { logDebug "current state.deviceProfile=${state.deviceProfile}, settings.forcedProfile=${settings?.forcedProfile}, getProfileKey()=${getProfileKey(settings?.forcedProfile)}" if (getProfileKey(settings?.forcedProfile) != state.deviceProfile) { logInfo "changing the device profile from ${state.deviceProfile} to ${getProfileKey(settings?.forcedProfile)}" state.deviceProfile = getProfileKey(settings?.forcedProfile) initializeVars(fullInit = false) resetPreferencesToDefaults(debug = true) logInfo 'press F5 to refresh the page' } } } /* groovylint-disable-next-line EmptyElseBlock */ else { logDebug "forcedProfile is not set" } final int interval = (settings?.pollingInterval as Integer) ?: 0 if (interval > 0) { logInfo "customUpdated: scheduling polling every ${interval} seconds" schedulePolling(interval) } else { unSchedulePolling() logInfo 'customUpdated: polling is disabled!' } // Itterates through all settings cmds += updateAllPreferences() // defined in deviceProfileLib sendZigbeeCommands(cmds) } /** * Schedule polling * @param intervalMins interval in seconds */ private void schedulePolling(final int intervalSecs) { String cron = getCron( intervalSecs ) logDebug "cron = ${cron}" schedule(cron, 'autoPoll') } private void unSchedulePolling() { unschedule('autoPoll') } /** * Scheduled job for polling device specific attribute(s) */ void autoPoll() { logDebug 'autoPoll()...' checkDriverVersion(state) List cmds = [] cmds = refreshFromDeviceProfileList() if (cmds != null && cmds != [] ) { sendZigbeeCommands(cmds) } } void customInitializeVars(final boolean fullInit=false) { logDebug "customInitializeVars(${fullInit})" if (state.deviceProfile == null || state.deviceProfile == '' || state.deviceProfile == 'UNKNOWN') { setDeviceNameAndProfile('TS0601', '_TZE200_v1jqz5cy') // in deviceProfileiLib.groovy } if (fullInit == true) { resetPreferencesToDefaults() } if (fullInit || settings?.pollingInterval == null) { device.updateSetting('pollingInterval', [value: PollingIntervalOpts.defaultValue.toString(), type: 'enum']) } } void customInitEvents(final boolean fullInit=false) { logDebug "customInitEvents()" } // https://github.com/dresden-elektronik/deconz-rest-plugin/blob/0107459aa42f8ac5333c67f415e2482069e4ff79/device_access_fn.cpp#L825 void customParseZdoClusters(Map descMap) { if (descMap.clusterInt == 0x0013) { logDebug "customParseZdoClusters() - device announce" sendZigbeeCommands(refreshQueryAllTuyaDP()) } } List refreshQueryAllTuyaDP() { return queryAllTuyaDP() } void test(String par) { long startTime = now() logDebug "test() started at ${startTime}" //parse('catchall: 0104 EF00 01 01 0040 00 7770 01 00 0000 02 01 00556701000100') def parpar = 'catchall: 0104 EF00 01 01 0040 00 7770 01 00 0000 02 01 00556701000100' for (int i=0; i<100; i++) { testFunc(parpar) } long endTime = now() logDebug "test() ended at ${endTime} (duration ${endTime - startTime}ms)" } // /////////////////////////////////////////////////////////////////// Libraries ////////////////////////////////////////////////////////////////////// // ~~~~~ start include (142) kkossev.deviceProfileLib ~~~~~ /* groovylint-disable CompileStatic, CouldBeSwitchStatement, DuplicateListLiteral, DuplicateNumberLiteral, DuplicateStringLiteral, ImplicitClosureParameter, ImplicitReturnStatement, Instanceof, LineLength, MethodCount, MethodSize, NestedBlockDepth, NoDouble, NoFloat, NoWildcardImports, ParameterName, PublicMethodsBeforeNonPublicMethods, UnnecessaryElseStatement, UnnecessaryGetter, UnnecessaryPublicModifier, UnnecessarySetter, UnusedImport */ // library marker kkossev.deviceProfileLib, line 1 library( // library marker kkossev.deviceProfileLib, line 2 base: 'driver', author: 'Krassimir Kossev', category: 'zigbee', description: 'Device Profile Library', name: 'deviceProfileLib', namespace: 'kkossev', // library marker kkossev.deviceProfileLib, line 3 importUrl: 'https://raw.githubusercontent.com/kkossev/Hubitat/refs/heads/development/Libraries/deviceProfileLib.groovy', documentationLink: 'https://github.com/kkossev/Hubitat/wiki/libraries-deviceProfileLib', // library marker kkossev.deviceProfileLib, line 4 version: '3.4.2' // library marker kkossev.deviceProfileLib, line 5 ) // library marker kkossev.deviceProfileLib, line 6 /* // library marker kkossev.deviceProfileLib, line 7 * Device Profile Library // library marker kkossev.deviceProfileLib, line 8 * // library marker kkossev.deviceProfileLib, line 9 * Licensed Virtual the Apache License, Version 2.0 (the "License"); you may not use this file except // library marker kkossev.deviceProfileLib, line 10 * in compliance with the License. You may obtain a copy of the License at: // library marker kkossev.deviceProfileLib, line 11 * // library marker kkossev.deviceProfileLib, line 12 * http://www.apache.org/licenses/LICENSE-2.0 // library marker kkossev.deviceProfileLib, line 13 * // library marker kkossev.deviceProfileLib, line 14 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed // library marker kkossev.deviceProfileLib, line 15 * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License // library marker kkossev.deviceProfileLib, line 16 * for the specific language governing permissions and limitations under the License. // library marker kkossev.deviceProfileLib, line 17 * // library marker kkossev.deviceProfileLib, line 18 * ver. 1.0.0 2023-11-04 kkossev - added deviceProfileLib (based on Tuya 4 In 1 driver) // library marker kkossev.deviceProfileLib, line 19 * ver. 3.0.0 2023-11-27 kkossev - fixes for use with commonLib; added processClusterAttributeFromDeviceProfile() method; added validateAndFixPreferences() method; inputIt bug fix; signedInt Preproc method; // library marker kkossev.deviceProfileLib, line 20 * ver. 3.0.1 2023-12-02 kkossev - release candidate // library marker kkossev.deviceProfileLib, line 21 * ver. 3.0.2 2023-12-17 kkossev - inputIt moved to the preferences section; setfunction replaced by customSetFunction; Groovy Linting; // library marker kkossev.deviceProfileLib, line 22 * ver. 3.0.4 2024-03-30 kkossev - more Groovy Linting; processClusterAttributeFromDeviceProfile exception fix; // library marker kkossev.deviceProfileLib, line 23 * ver. 3.1.0 2024-04-03 kkossev - more Groovy Linting; deviceProfilesV3, enum pars bug fix; // library marker kkossev.deviceProfileLib, line 24 * ver. 3.1.1 2024-04-21 kkossev - deviceProfilesV3 bug fix; tuyaDPs list of maps bug fix; resetPreferencesToDefaults bug fix; // library marker kkossev.deviceProfileLib, line 25 * ver. 3.1.2 2024-05-05 kkossev - added isSpammyDeviceProfile() // library marker kkossev.deviceProfileLib, line 26 * ver. 3.1.3 2024-05-21 kkossev - skip processClusterAttributeFromDeviceProfile if cluster or attribute or value is missing // library marker kkossev.deviceProfileLib, line 27 * ver. 3.2.0 2024-05-25 kkossev - commonLib 3.2.0 allignment; // library marker kkossev.deviceProfileLib, line 28 * ver. 3.2.1 2024-06-06 kkossev - Tuya Multi Sensor 4 In 1 (V3) driver allignment (customProcessDeviceProfileEvent); getDeviceProfilesMap bug fix; forcedProfile is always shown in preferences; // library marker kkossev.deviceProfileLib, line 29 * ver. 3.3.0 2024-06-29 kkossev - empty preferences bug fix; zclWriteAttribute delay 50 ms; added advanced check in inputIt(); fixed 'Cannot get property 'rw' on null object' bug; fixed enum attributes first event numeric value bug; // library marker kkossev.deviceProfileLib, line 30 * ver. 3.3.1 2024-07-06 kkossev - added powerSource event in the initEventsDeviceProfile // library marker kkossev.deviceProfileLib, line 31 * ver. 3.3.2 2024-08-18 kkossev - release 3.3.2 // library marker kkossev.deviceProfileLib, line 32 * ver. 3.3.3 2024-08-18 kkossev - sendCommand and setPar commands commented out; must be declared in the main driver where really needed // library marker kkossev.deviceProfileLib, line 33 * ver. 3.3.4 2024-09-28 kkossev - fixed exceptions in resetPreferencesToDefaults() and initEventsDeviceProfile() // library marker kkossev.deviceProfileLib, line 34 * ver. 3.4.0 2025-02-02 kkossev - deviceProfilesV3 optimizations (defaultFingerprint); is2in1() mod // library marker kkossev.deviceProfileLib, line 35 * ver. 3.4.1 2025-02-02 kkossev - setPar help improvements; // library marker kkossev.deviceProfileLib, line 36 * ver. 3.4.2 2025-03-24 kkossev - added refreshFromConfigureReadList() method; documentation update; getDeviceNameAndProfile uses DEVICE.description instead of deviceJoinName // library marker kkossev.deviceProfileLib, line 37 * ver. 3.4.3 2025-04-25 kkossev - HE platfrom version 2.4.1.x decimal preferences patch/workaround. // library marker kkossev.deviceProfileLib, line 38 * // library marker kkossev.deviceProfileLib, line 39 * TODO - remove the 2-in-1 patch ! // library marker kkossev.deviceProfileLib, line 40 * TODO - add updateStateUnknownDPs (from the 4-in-1 driver) // library marker kkossev.deviceProfileLib, line 41 * TODO - when [refresh], send Info logs for parameters that are not events or preferences // library marker kkossev.deviceProfileLib, line 42 * TODO: refactor sendAttribute ! sendAttribute exception bug fix for virtual devices; check if String getObjectClassName(Object o) is in 2.3.3.137, can be used? // library marker kkossev.deviceProfileLib, line 43 * TODO: add _DEBUG command (for temporary switching the debug logs on/off) // library marker kkossev.deviceProfileLib, line 44 * TODO: allow NULL parameters default values in the device profiles // library marker kkossev.deviceProfileLib, line 45 * TODO: handle preferences of a type TEXT // library marker kkossev.deviceProfileLib, line 46 * // library marker kkossev.deviceProfileLib, line 47 */ // library marker kkossev.deviceProfileLib, line 48 static String deviceProfileLibVersion() { '3.4.3' } // library marker kkossev.deviceProfileLib, line 50 static String deviceProfileLibStamp() { '2025/04/25 12:43 PM' } // library marker kkossev.deviceProfileLib, line 51 import groovy.json.* // library marker kkossev.deviceProfileLib, line 52 import groovy.transform.Field // library marker kkossev.deviceProfileLib, line 53 import hubitat.zigbee.clusters.iaszone.ZoneStatus // library marker kkossev.deviceProfileLib, line 54 import hubitat.zigbee.zcl.DataType // library marker kkossev.deviceProfileLib, line 55 import java.util.concurrent.ConcurrentHashMap // library marker kkossev.deviceProfileLib, line 56 import groovy.transform.CompileStatic // library marker kkossev.deviceProfileLib, line 58 metadata { // library marker kkossev.deviceProfileLib, line 60 // no capabilities // library marker kkossev.deviceProfileLib, line 61 // no attributes // library marker kkossev.deviceProfileLib, line 62 /* // library marker kkossev.deviceProfileLib, line 63 // copy the following commands to the main driver, if needed // library marker kkossev.deviceProfileLib, line 64 command 'sendCommand', [ // library marker kkossev.deviceProfileLib, line 65 [name:'command', type: 'STRING', description: 'command name', constraints: ['STRING']], // library marker kkossev.deviceProfileLib, line 66 [name:'val', type: 'STRING', description: 'command parameter value', constraints: ['STRING']] // library marker kkossev.deviceProfileLib, line 67 ] // library marker kkossev.deviceProfileLib, line 68 command 'setPar', [ // library marker kkossev.deviceProfileLib, line 69 [name:'par', type: 'STRING', description: 'preference parameter name', constraints: ['STRING']], // library marker kkossev.deviceProfileLib, line 70 [name:'val', type: 'STRING', description: 'preference parameter value', constraints: ['STRING']] // library marker kkossev.deviceProfileLib, line 71 ] // library marker kkossev.deviceProfileLib, line 72 */ // library marker kkossev.deviceProfileLib, line 73 preferences { // library marker kkossev.deviceProfileLib, line 74 if (device) { // library marker kkossev.deviceProfileLib, line 75 input(name: 'forcedProfile', type: 'enum', title: 'Device Profile', description: 'Manually change the Device Profile, if the model/manufacturer was not recognized automatically.
Warning! Manually setting a device profile may not always work!', options: getDeviceProfilesMap()) // library marker kkossev.deviceProfileLib, line 76 // itterate over DEVICE.preferences map and inputIt all // library marker kkossev.deviceProfileLib, line 77 if (DEVICE != null && DEVICE?.preferences != null && DEVICE?.preferences != [:] && DEVICE?.device?.isDepricated != true) { // library marker kkossev.deviceProfileLib, line 78 (DEVICE?.preferences).each { key, value -> // library marker kkossev.deviceProfileLib, line 79 Map inputMap = inputIt(key) // library marker kkossev.deviceProfileLib, line 80 if (inputMap != null && inputMap != [:]) { // library marker kkossev.deviceProfileLib, line 81 input inputMap // library marker kkossev.deviceProfileLib, line 82 } // library marker kkossev.deviceProfileLib, line 83 } // library marker kkossev.deviceProfileLib, line 84 } // library marker kkossev.deviceProfileLib, line 85 } // library marker kkossev.deviceProfileLib, line 86 } // library marker kkossev.deviceProfileLib, line 87 } // library marker kkossev.deviceProfileLib, line 88 private boolean is2in1() { return getDeviceProfile().startsWith('TS0601_2IN1') } // patch! // library marker kkossev.deviceProfileLib, line 90 public String getDeviceProfile() { state?.deviceProfile ?: 'UNKNOWN' } // library marker kkossev.deviceProfileLib, line 92 public Map getDEVICE() { deviceProfilesV3 != null ? deviceProfilesV3[getDeviceProfile()] : deviceProfilesV2 != null ? deviceProfilesV2[getDeviceProfile()] : [:] } // library marker kkossev.deviceProfileLib, line 93 public Set getDeviceProfiles() { deviceProfilesV3 != null ? deviceProfilesV3?.keySet() : deviceProfilesV2 != null ? deviceProfilesV2?.keySet() : [] } // library marker kkossev.deviceProfileLib, line 94 public List getDeviceProfilesMap() { // library marker kkossev.deviceProfileLib, line 96 if (deviceProfilesV3 == null) { // library marker kkossev.deviceProfileLib, line 97 if (deviceProfilesV2 == null) { return [] } // library marker kkossev.deviceProfileLib, line 98 return deviceProfilesV2.values().description as List // library marker kkossev.deviceProfileLib, line 99 } // library marker kkossev.deviceProfileLib, line 100 List activeProfiles = [] // library marker kkossev.deviceProfileLib, line 101 deviceProfilesV3.each { profileName, profileMap -> // library marker kkossev.deviceProfileLib, line 102 if ((profileMap.device?.isDepricated ?: false) != true) { // library marker kkossev.deviceProfileLib, line 103 activeProfiles.add(profileMap.description ?: '---') // library marker kkossev.deviceProfileLib, line 104 } // library marker kkossev.deviceProfileLib, line 105 } // library marker kkossev.deviceProfileLib, line 106 return activeProfiles // library marker kkossev.deviceProfileLib, line 107 } // library marker kkossev.deviceProfileLib, line 108 // ---------------------------------- deviceProfilesV3 helper functions -------------------------------------------- // library marker kkossev.deviceProfileLib, line 110 /** // library marker kkossev.deviceProfileLib, line 112 * Returns the profile key for a given profile description. // library marker kkossev.deviceProfileLib, line 113 * @param valueStr The profile description to search for. // library marker kkossev.deviceProfileLib, line 114 * @return The profile key if found, otherwise null. // library marker kkossev.deviceProfileLib, line 115 */ // library marker kkossev.deviceProfileLib, line 116 public String getProfileKey(final String valueStr) { // library marker kkossev.deviceProfileLib, line 117 if (deviceProfilesV3 != null) { return deviceProfilesV3.find { _, profileMap -> profileMap.description == valueStr }?.key } // library marker kkossev.deviceProfileLib, line 118 else if (deviceProfilesV2 != null) { return deviceProfilesV2.find { _, profileMap -> profileMap.description == valueStr }?.key } // library marker kkossev.deviceProfileLib, line 119 else { return null } // library marker kkossev.deviceProfileLib, line 120 } // library marker kkossev.deviceProfileLib, line 121 /** // library marker kkossev.deviceProfileLib, line 123 * Finds the preferences map for the given parameter. // library marker kkossev.deviceProfileLib, line 124 * @param param The parameter to find the preferences map for. // library marker kkossev.deviceProfileLib, line 125 * @param debug Whether or not to output debug logs. // library marker kkossev.deviceProfileLib, line 126 * @return returns either tuyaDPs or attributes map, depending on where the preference (param) is found // library marker kkossev.deviceProfileLib, line 127 * @return empty map [:] if param is not defined for this device. // library marker kkossev.deviceProfileLib, line 128 */ // library marker kkossev.deviceProfileLib, line 129 private Map getPreferencesMapByName(final String param, boolean debug=false) { // library marker kkossev.deviceProfileLib, line 130 Map foundMap = [:] // library marker kkossev.deviceProfileLib, line 131 if (!(param in DEVICE?.preferences)) { if (debug) { log.warn "getPreferencesMapByName: preference ${param} not defined for this device!" } ; return [:] } // library marker kkossev.deviceProfileLib, line 132 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 133 def preference // library marker kkossev.deviceProfileLib, line 134 try { // library marker kkossev.deviceProfileLib, line 135 preference = DEVICE?.preferences["$param"] // library marker kkossev.deviceProfileLib, line 136 if (debug) { log.debug "getPreferencesMapByName: preference ${param} found. value is ${preference}" } // library marker kkossev.deviceProfileLib, line 137 if (preference in [true, false]) { // library marker kkossev.deviceProfileLib, line 138 // find the preference in the tuyaDPs map // library marker kkossev.deviceProfileLib, line 139 logDebug "getPreferencesMapByName: preference ${param} is boolean" // library marker kkossev.deviceProfileLib, line 140 return [:] // no maps for predefined preferences ! // library marker kkossev.deviceProfileLib, line 141 } // library marker kkossev.deviceProfileLib, line 142 if (safeToInt(preference, -1) > 0) { //if (preference instanceof Number) { // library marker kkossev.deviceProfileLib, line 143 int dp = safeToInt(preference) // library marker kkossev.deviceProfileLib, line 144 //if (debug) log.trace "getPreferencesMapByName: param ${param} preference ${preference} is number (${dp})" // library marker kkossev.deviceProfileLib, line 145 foundMap = DEVICE?.tuyaDPs.find { it.dp == dp } // library marker kkossev.deviceProfileLib, line 146 } // library marker kkossev.deviceProfileLib, line 147 else { // cluster:attribute // library marker kkossev.deviceProfileLib, line 148 //if (debug) { log.trace "${DEVICE?.attributes}" } // library marker kkossev.deviceProfileLib, line 149 foundMap = DEVICE?.attributes.find { it.at == preference } // library marker kkossev.deviceProfileLib, line 150 } // library marker kkossev.deviceProfileLib, line 151 // TODO - could be also 'true' or 'false' ... // library marker kkossev.deviceProfileLib, line 152 } catch (e) { // library marker kkossev.deviceProfileLib, line 153 if (debug) { log.warn "getPreferencesMapByName: exception ${e} caught when getting preference ${param} !" } // library marker kkossev.deviceProfileLib, line 154 return [:] // library marker kkossev.deviceProfileLib, line 155 } // library marker kkossev.deviceProfileLib, line 156 if (debug) { log.debug "getPreferencesMapByName: foundMap = ${foundMap}" } // library marker kkossev.deviceProfileLib, line 157 return foundMap // library marker kkossev.deviceProfileLib, line 158 } // library marker kkossev.deviceProfileLib, line 159 public Map getAttributesMap(String attribName, boolean debug=false) { // library marker kkossev.deviceProfileLib, line 161 Map foundMap = [:] // library marker kkossev.deviceProfileLib, line 162 List searchMapList = [] // library marker kkossev.deviceProfileLib, line 163 if (debug) { logDebug "getAttributesMap: searching for attribute ${attribName} in tuyaDPs" } // library marker kkossev.deviceProfileLib, line 164 if (DEVICE?.tuyaDPs != null && DEVICE?.tuyaDPs != [:]) { // library marker kkossev.deviceProfileLib, line 165 searchMapList = DEVICE?.tuyaDPs // library marker kkossev.deviceProfileLib, line 166 foundMap = searchMapList.find { it.name == attribName } // library marker kkossev.deviceProfileLib, line 167 if (foundMap != null) { // library marker kkossev.deviceProfileLib, line 168 if (debug) { logDebug "getAttributesMap: foundMap = ${foundMap}" } // library marker kkossev.deviceProfileLib, line 169 return foundMap // library marker kkossev.deviceProfileLib, line 170 } // library marker kkossev.deviceProfileLib, line 171 } // library marker kkossev.deviceProfileLib, line 172 if (debug) { logDebug "getAttributesMap: searching for attribute ${attribName} in attributes" } // library marker kkossev.deviceProfileLib, line 173 if (DEVICE?.attributes != null && DEVICE?.attributes != [:]) { // library marker kkossev.deviceProfileLib, line 174 searchMapList = DEVICE?.attributes // library marker kkossev.deviceProfileLib, line 175 foundMap = searchMapList.find { it.name == attribName } // library marker kkossev.deviceProfileLib, line 176 if (foundMap != null) { // library marker kkossev.deviceProfileLib, line 177 if (debug) { logDebug "getAttributesMap: foundMap = ${foundMap}" } // library marker kkossev.deviceProfileLib, line 178 return foundMap // library marker kkossev.deviceProfileLib, line 179 } // library marker kkossev.deviceProfileLib, line 180 } // library marker kkossev.deviceProfileLib, line 181 if (debug) { logDebug "getAttributesMap: attribute ${attribName} not found in tuyaDPs or attributes map! foundMap=${foundMap}" } // library marker kkossev.deviceProfileLib, line 182 return [:] // library marker kkossev.deviceProfileLib, line 183 } // library marker kkossev.deviceProfileLib, line 184 /** // library marker kkossev.deviceProfileLib, line 186 * Resets the device preferences to their default values. // library marker kkossev.deviceProfileLib, line 187 * @param debug A boolean indicating whether to output debug information. // library marker kkossev.deviceProfileLib, line 188 */ // library marker kkossev.deviceProfileLib, line 189 public void resetPreferencesToDefaults(boolean debug=false) { // library marker kkossev.deviceProfileLib, line 190 logDebug "resetPreferencesToDefaults: DEVICE=${DEVICE?.description} preferences=${DEVICE?.preferences}" // library marker kkossev.deviceProfileLib, line 191 if (DEVICE == null || DEVICE?.preferences == null || DEVICE?.preferences == [:]) { logDebug 'Preferences not found!' ; return } // library marker kkossev.deviceProfileLib, line 192 Map preferences = DEVICE?.preferences ?: [:] // library marker kkossev.deviceProfileLib, line 193 if (preferences == null || preferences == [:]) { logDebug 'Preferences not found!' ; return } // library marker kkossev.deviceProfileLib, line 194 Map parMap = [:] // library marker kkossev.deviceProfileLib, line 195 preferences.each { parName, mapValue -> // library marker kkossev.deviceProfileLib, line 196 if (debug) { log.trace "$parName $mapValue" } // library marker kkossev.deviceProfileLib, line 197 if ((mapValue in [true, false]) || (mapValue in ['true', 'false'])) { // library marker kkossev.deviceProfileLib, line 198 logDebug "Preference ${parName} is predefined -> (${mapValue})" // what was the idea here? // library marker kkossev.deviceProfileLib, line 199 return // continue // library marker kkossev.deviceProfileLib, line 200 } // library marker kkossev.deviceProfileLib, line 201 parMap = getPreferencesMapByName(parName, false) // the individual preference map // library marker kkossev.deviceProfileLib, line 202 if (parMap == null || parMap?.isEmpty()) { logDebug "Preference ${parName} not found in tuyaDPs or attributes map!"; return } // continue // library marker kkossev.deviceProfileLib, line 203 // at:'0x0406:0x0020', name:'fadingTime', type:'enum', dt: '0x21', rw: 'rw', min:15, max:999, defVal:'30', scale:1, unit:'seconds', map:[15:'15 seconds', 30:'30 seconds', 60:'60 seconds', 120:'120 seconds', 300:'300 seconds'], title:'Fading Time', description:'Radar fading time in seconds'], // library marker kkossev.deviceProfileLib, line 204 if (parMap?.defVal == null) { logDebug "no default value for preference ${parName} !" ; return } // continue // library marker kkossev.deviceProfileLib, line 205 if (debug) { log.info "setting par ${parMap.name} defVal = ${parMap.defVal} (type:${parMap.type})" } // library marker kkossev.deviceProfileLib, line 206 String str = parMap.name // library marker kkossev.deviceProfileLib, line 207 device.updateSetting("$str", [value:parMap.defVal as String, type:parMap.type]) // library marker kkossev.deviceProfileLib, line 208 } // library marker kkossev.deviceProfileLib, line 209 logInfo 'Preferences reset to default values' // library marker kkossev.deviceProfileLib, line 210 } // library marker kkossev.deviceProfileLib, line 211 /** // library marker kkossev.deviceProfileLib, line 213 * Returns a list of valid parameters per model based on the device preferences. // library marker kkossev.deviceProfileLib, line 214 * // library marker kkossev.deviceProfileLib, line 215 * @return List of valid parameters. // library marker kkossev.deviceProfileLib, line 216 */ // library marker kkossev.deviceProfileLib, line 217 private List getValidParsPerModel() { // library marker kkossev.deviceProfileLib, line 218 List validPars = [] // library marker kkossev.deviceProfileLib, line 219 if (DEVICE?.preferences != null && DEVICE?.preferences != [:]) { // library marker kkossev.deviceProfileLib, line 220 // use the preferences to validate the parameters // library marker kkossev.deviceProfileLib, line 221 validPars = DEVICE?.preferences.keySet().toList() // library marker kkossev.deviceProfileLib, line 222 } // library marker kkossev.deviceProfileLib, line 223 return validPars // library marker kkossev.deviceProfileLib, line 224 } // library marker kkossev.deviceProfileLib, line 225 /* groovylint-disable-next-line MethodReturnTypeRequired, NoDef */ // library marker kkossev.deviceProfileLib, line 227 private def getScaledPreferenceValue(String preference, Map dpMap) { // TODO - not used ??? // library marker kkossev.deviceProfileLib, line 228 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 229 def value = settings."${preference}" // library marker kkossev.deviceProfileLib, line 230 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 231 def scaledValue // library marker kkossev.deviceProfileLib, line 232 if (value == null) { // library marker kkossev.deviceProfileLib, line 233 logDebug "getScaledPreferenceValue: preference ${preference} not found!" // library marker kkossev.deviceProfileLib, line 234 return null // library marker kkossev.deviceProfileLib, line 235 } // library marker kkossev.deviceProfileLib, line 236 switch (dpMap.type) { // library marker kkossev.deviceProfileLib, line 237 case 'number' : // library marker kkossev.deviceProfileLib, line 238 scaledValue = safeToInt(value) // library marker kkossev.deviceProfileLib, line 239 break // library marker kkossev.deviceProfileLib, line 240 case 'decimal' : // library marker kkossev.deviceProfileLib, line 241 scaledValue = safeToDouble(value) // library marker kkossev.deviceProfileLib, line 242 if (dpMap.scale != null && dpMap.scale != 1) { // library marker kkossev.deviceProfileLib, line 243 scaledValue = Math.round(scaledValue * dpMap.scale) // library marker kkossev.deviceProfileLib, line 244 } // library marker kkossev.deviceProfileLib, line 245 break // library marker kkossev.deviceProfileLib, line 246 case 'bool' : // library marker kkossev.deviceProfileLib, line 247 scaledValue = value == 'true' ? 1 : 0 // library marker kkossev.deviceProfileLib, line 248 break // library marker kkossev.deviceProfileLib, line 249 case 'enum' : // library marker kkossev.deviceProfileLib, line 250 //logWarn "getScaledPreferenceValue: ENUM preference ${preference} type:${dpMap.type} value = ${value} dpMap.scale=${dpMap.scale}" // library marker kkossev.deviceProfileLib, line 251 if (dpMap.map == null) { // library marker kkossev.deviceProfileLib, line 252 logDebug "getScaledPreferenceValue: preference ${preference} has no map defined!" // library marker kkossev.deviceProfileLib, line 253 return null // library marker kkossev.deviceProfileLib, line 254 } // library marker kkossev.deviceProfileLib, line 255 scaledValue = value // library marker kkossev.deviceProfileLib, line 256 if (dpMap.scale != null && safeToInt(dpMap.scale) != 1) { // library marker kkossev.deviceProfileLib, line 257 scaledValue = Math.round(safeToDouble(scaledValue ) * safeToInt(dpMap.scale)) // library marker kkossev.deviceProfileLib, line 258 } // library marker kkossev.deviceProfileLib, line 259 break // library marker kkossev.deviceProfileLib, line 260 default : // library marker kkossev.deviceProfileLib, line 261 logDebug "getScaledPreferenceValue: preference ${preference} has unsupported type ${dpMap.type}!" // library marker kkossev.deviceProfileLib, line 262 return null // library marker kkossev.deviceProfileLib, line 263 } // library marker kkossev.deviceProfileLib, line 264 //logDebug "getScaledPreferenceValue: preference ${preference} value = ${value} scaledValue = ${scaledValue} (scale=${dpMap.scale})" // library marker kkossev.deviceProfileLib, line 265 return scaledValue // library marker kkossev.deviceProfileLib, line 266 } // library marker kkossev.deviceProfileLib, line 267 // called from customUpdated() method in the custom driver // library marker kkossev.deviceProfileLib, line 269 // TODO !!!!!!!!!! - refactor it !!! IAS settings do not use Tuya DPs !!! // library marker kkossev.deviceProfileLib, line 270 public void updateAllPreferences() { // library marker kkossev.deviceProfileLib, line 271 logDebug "updateAllPreferences: preferences=${DEVICE?.preferences}" // library marker kkossev.deviceProfileLib, line 272 if (DEVICE?.preferences == null || DEVICE?.preferences == [:]) { // library marker kkossev.deviceProfileLib, line 273 logDebug "updateAllPreferences: no preferences defined for device profile ${getDeviceProfile()}" // library marker kkossev.deviceProfileLib, line 274 return // library marker kkossev.deviceProfileLib, line 275 } // library marker kkossev.deviceProfileLib, line 276 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 277 def preferenceValue // int or String for enums // library marker kkossev.deviceProfileLib, line 278 // itterate over the preferences map and update the device settings // library marker kkossev.deviceProfileLib, line 279 (DEVICE?.preferences).each { name, dp -> // library marker kkossev.deviceProfileLib, line 280 Map foundMap = getPreferencesMapByName(name, false) // library marker kkossev.deviceProfileLib, line 281 logDebug "updateAllPreferences: foundMap = ${foundMap}" // library marker kkossev.deviceProfileLib, line 282 if (foundMap != null && foundMap != [:]) { // library marker kkossev.deviceProfileLib, line 283 // preferenceValue = getScaledPreferenceValue(name, foundMap) // library marker kkossev.deviceProfileLib, line 284 preferenceValue = settings."${name}" // library marker kkossev.deviceProfileLib, line 285 logTrace"preferenceValue = ${preferenceValue}" // library marker kkossev.deviceProfileLib, line 286 if (foundMap.type == 'enum' && foundMap.scale != null && foundMap.scale != 1 && foundMap.scale != 0) { // library marker kkossev.deviceProfileLib, line 287 // scale the value // library marker kkossev.deviceProfileLib, line 288 preferenceValue = (safeToDouble(preferenceValue) / safeToInt(foundMap.scale)) as double // library marker kkossev.deviceProfileLib, line 289 } // library marker kkossev.deviceProfileLib, line 290 if (preferenceValue != null) { // library marker kkossev.deviceProfileLib, line 291 setPar(name, preferenceValue.toString()) // library marker kkossev.deviceProfileLib, line 292 } // library marker kkossev.deviceProfileLib, line 293 else { logDebug "updateAllPreferences: preference ${name} is not set (preferenceValue was null)" ; return } // library marker kkossev.deviceProfileLib, line 294 } // library marker kkossev.deviceProfileLib, line 295 else { logDebug "warning: couldn't find map for preference ${name}" ; return } // TODO - supress the warning if the preference was boolean true/false // library marker kkossev.deviceProfileLib, line 296 } // library marker kkossev.deviceProfileLib, line 297 return // library marker kkossev.deviceProfileLib, line 298 } // library marker kkossev.deviceProfileLib, line 299 /* groovylint-disable-next-line MethodReturnTypeRequired, NoDef */ // library marker kkossev.deviceProfileLib, line 301 def divideBy100(int val) { return (val as int) / 100 } // library marker kkossev.deviceProfileLib, line 302 int multiplyBy100(int val) { return (val as int) * 100 } // library marker kkossev.deviceProfileLib, line 303 int divideBy10(int val) { // library marker kkossev.deviceProfileLib, line 304 if (val > 10) { return (val as int) / 10 } // library marker kkossev.deviceProfileLib, line 305 else { return (val as int) } // library marker kkossev.deviceProfileLib, line 306 } // library marker kkossev.deviceProfileLib, line 307 int multiplyBy10(int val) { return (val as int) * 10 } // library marker kkossev.deviceProfileLib, line 308 int divideBy1(int val) { return (val as int) / 1 } //tests // library marker kkossev.deviceProfileLib, line 309 int signedInt(int val) { // library marker kkossev.deviceProfileLib, line 310 if (val > 127) { return (val as int) - 256 } // library marker kkossev.deviceProfileLib, line 311 else { return (val as int) } // library marker kkossev.deviceProfileLib, line 312 } // library marker kkossev.deviceProfileLib, line 313 int invert(int val) { // library marker kkossev.deviceProfileLib, line 314 if (settings.invertMotion == true) { return val == 0 ? 1 : 0 } // library marker kkossev.deviceProfileLib, line 315 else { return val } // library marker kkossev.deviceProfileLib, line 316 } // library marker kkossev.deviceProfileLib, line 317 // called from setPar and sendAttribite methods for non-Tuya DPs // library marker kkossev.deviceProfileLib, line 319 private List zclWriteAttribute(Map attributesMap, int scaledValue) { // library marker kkossev.deviceProfileLib, line 320 if (attributesMap == null || attributesMap == [:]) { logWarn "attributesMap=${attributesMap}" ; return [] } // library marker kkossev.deviceProfileLib, line 321 List cmds = [] // library marker kkossev.deviceProfileLib, line 322 Map map = [:] // library marker kkossev.deviceProfileLib, line 323 // cluster:attribute // library marker kkossev.deviceProfileLib, line 324 try { // library marker kkossev.deviceProfileLib, line 325 map['cluster'] = hubitat.helper.HexUtils.hexStringToInt((attributesMap.at).split(':')[0]) as Integer // library marker kkossev.deviceProfileLib, line 326 map['attribute'] = hubitat.helper.HexUtils.hexStringToInt((attributesMap.at).split(':')[1]) as Integer // library marker kkossev.deviceProfileLib, line 327 map['dt'] = (attributesMap.dt != null && attributesMap.dt != '') ? hubitat.helper.HexUtils.hexStringToInt(attributesMap.dt) as Integer : null // library marker kkossev.deviceProfileLib, line 328 map['mfgCode'] = attributesMap.mfgCode ? attributesMap.mfgCode as String : null // library marker kkossev.deviceProfileLib, line 329 } // library marker kkossev.deviceProfileLib, line 330 catch (e) { logWarn "setPar: Exception caught while splitting cluser and attribute $customSetFunction($scaledValue) (val=${val})) : '${e}' " ; return [] } // library marker kkossev.deviceProfileLib, line 331 // dt (data type) is obligatory when writing to a cluster... // library marker kkossev.deviceProfileLib, line 332 if (attributesMap.rw != null && attributesMap.rw == 'rw' && map.dt == null) { // library marker kkossev.deviceProfileLib, line 333 map.dt = attributesMap.type in ['number', 'decimal'] ? DataType.INT16 : DataType.ENUM8 // library marker kkossev.deviceProfileLib, line 334 logDebug "cluster:attribute ${attributesMap.at} is read-write, but no data type (dt) is defined! Assuming 0x${zigbee.convertToHexString(map.dt, 2)}" // library marker kkossev.deviceProfileLib, line 335 } // library marker kkossev.deviceProfileLib, line 336 if (map.mfgCode != null && map.mfgCode != '') { // library marker kkossev.deviceProfileLib, line 337 Map mfgCode = map.mfgCode != null ? ['mfgCode':map.mfgCode] : [:] // library marker kkossev.deviceProfileLib, line 338 cmds = zigbee.writeAttribute(map.cluster as int, map.attribute as int, map.dt as int, scaledValue, mfgCode, delay = 50) // library marker kkossev.deviceProfileLib, line 339 } // library marker kkossev.deviceProfileLib, line 340 else { // library marker kkossev.deviceProfileLib, line 341 cmds = zigbee.writeAttribute(map.cluster as int, map.attribute as int, map.dt as int, scaledValue, [:], delay = 50) // library marker kkossev.deviceProfileLib, line 342 } // library marker kkossev.deviceProfileLib, line 343 return cmds // library marker kkossev.deviceProfileLib, line 344 } // library marker kkossev.deviceProfileLib, line 345 /** // library marker kkossev.deviceProfileLib, line 347 * Called from setPar() method only! // library marker kkossev.deviceProfileLib, line 348 * Validates the parameter value based on the given dpMap type and scales it if needed. // library marker kkossev.deviceProfileLib, line 349 * // library marker kkossev.deviceProfileLib, line 350 * @param dpMap The map containing the parameter type, minimum and maximum values. // library marker kkossev.deviceProfileLib, line 351 * @param val The value to be validated and scaled. // library marker kkossev.deviceProfileLib, line 352 * @return The validated and scaled value if it is within the specified range, null otherwise. // library marker kkossev.deviceProfileLib, line 353 */ // library marker kkossev.deviceProfileLib, line 354 /* groovylint-disable-next-line MethodReturnTypeRequired, NoDef */ // library marker kkossev.deviceProfileLib, line 355 private def validateAndScaleParameterValue(Map dpMap, String val) { // library marker kkossev.deviceProfileLib, line 356 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 357 def value // validated value - integer, floar // library marker kkossev.deviceProfileLib, line 358 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 359 def scaledValue // // library marker kkossev.deviceProfileLib, line 360 //logDebug "validateAndScaleParameterValue: dpMap=${dpMap} val=${val}" // library marker kkossev.deviceProfileLib, line 361 switch (dpMap.type) { // library marker kkossev.deviceProfileLib, line 362 case 'number' : // library marker kkossev.deviceProfileLib, line 363 // TODO - negative values ! // library marker kkossev.deviceProfileLib, line 364 // TODO - better conversion to integer! // library marker kkossev.deviceProfileLib, line 365 value = safeToInt(val, 0) // library marker kkossev.deviceProfileLib, line 366 //scaledValue = value // library marker kkossev.deviceProfileLib, line 367 // scale the value - added 10/26/2023 also for integer values ! // library marker kkossev.deviceProfileLib, line 368 if (dpMap.scale != null) { // library marker kkossev.deviceProfileLib, line 369 scaledValue = (value * dpMap.scale) as Integer // library marker kkossev.deviceProfileLib, line 370 } // library marker kkossev.deviceProfileLib, line 371 else { // library marker kkossev.deviceProfileLib, line 372 scaledValue = value // library marker kkossev.deviceProfileLib, line 373 } // library marker kkossev.deviceProfileLib, line 374 break // library marker kkossev.deviceProfileLib, line 375 case 'decimal' : // library marker kkossev.deviceProfileLib, line 377 value = safeToDouble(val, 0.0) // library marker kkossev.deviceProfileLib, line 378 // scale the value // library marker kkossev.deviceProfileLib, line 379 if (dpMap.scale != null) { // library marker kkossev.deviceProfileLib, line 380 scaledValue = (value * dpMap.scale) as Integer // library marker kkossev.deviceProfileLib, line 381 } // library marker kkossev.deviceProfileLib, line 382 else { // library marker kkossev.deviceProfileLib, line 383 scaledValue = value // library marker kkossev.deviceProfileLib, line 384 } // library marker kkossev.deviceProfileLib, line 385 break // library marker kkossev.deviceProfileLib, line 386 case 'bool' : // library marker kkossev.deviceProfileLib, line 388 if (val == '0' || val == 'false') { value = scaledValue = 0 } // library marker kkossev.deviceProfileLib, line 389 else if (val == '1' || val == 'true') { value = scaledValue = 1 } // library marker kkossev.deviceProfileLib, line 390 else { // library marker kkossev.deviceProfileLib, line 391 logInfo "bool parameter ${val}. value must be one of 0 1 false true" // library marker kkossev.deviceProfileLib, line 392 return null // library marker kkossev.deviceProfileLib, line 393 } // library marker kkossev.deviceProfileLib, line 394 break // library marker kkossev.deviceProfileLib, line 395 case 'enum' : // library marker kkossev.deviceProfileLib, line 396 // enums are always integer values // library marker kkossev.deviceProfileLib, line 397 // check if the scaling is different than 1 in dpMap // library marker kkossev.deviceProfileLib, line 398 logTrace "validateAndScaleParameterValue: enum parameter ${val}. dpMap=${dpMap}" // library marker kkossev.deviceProfileLib, line 399 Integer scale = safeToInt(dpMap.scale) // library marker kkossev.deviceProfileLib, line 400 if (scale != null && scale != 0 && scale != 1) { // library marker kkossev.deviceProfileLib, line 401 // we have a float parameter input - convert it to int // library marker kkossev.deviceProfileLib, line 402 value = safeToDouble(val, -1.0) // library marker kkossev.deviceProfileLib, line 403 scaledValue = (value * safeToInt(dpMap.scale)) as Integer // library marker kkossev.deviceProfileLib, line 404 } // library marker kkossev.deviceProfileLib, line 405 else { // library marker kkossev.deviceProfileLib, line 406 value = scaledValue = safeToInt(val, -1) // library marker kkossev.deviceProfileLib, line 407 } // library marker kkossev.deviceProfileLib, line 408 if (scaledValue == null || scaledValue < 0) { // library marker kkossev.deviceProfileLib, line 409 // get the keys of dpMap.map as a List // library marker kkossev.deviceProfileLib, line 410 //List keys = dpMap.map.keySet().toList() // library marker kkossev.deviceProfileLib, line 411 //logDebug "${device.displayName} validateAndScaleParameterValue: enum parameter ${val}. value must be one of ${keys}" // library marker kkossev.deviceProfileLib, line 412 // find the key for the value // library marker kkossev.deviceProfileLib, line 413 String key = dpMap.map.find { it.value == val }?.key // library marker kkossev.deviceProfileLib, line 414 logTrace "validateAndScaleParameterValue: enum parameter ${val}. key=${key}" // library marker kkossev.deviceProfileLib, line 415 if (key == null) { // library marker kkossev.deviceProfileLib, line 416 logInfo "invalid enum parameter ${val}. value must be one of ${dpMap.map}" // library marker kkossev.deviceProfileLib, line 417 return null // library marker kkossev.deviceProfileLib, line 418 } // library marker kkossev.deviceProfileLib, line 419 value = scaledValue = key as Integer // library marker kkossev.deviceProfileLib, line 420 //return null // library marker kkossev.deviceProfileLib, line 421 } // library marker kkossev.deviceProfileLib, line 422 break // library marker kkossev.deviceProfileLib, line 423 default : // library marker kkossev.deviceProfileLib, line 424 logWarn "validateAndScaleParameterValue: unsupported dpMap type ${parType}" // library marker kkossev.deviceProfileLib, line 425 return null // library marker kkossev.deviceProfileLib, line 426 } // library marker kkossev.deviceProfileLib, line 427 //logTrace "validateAndScaleParameterValue before checking scaledValue=${scaledValue}" // library marker kkossev.deviceProfileLib, line 428 // check if the value is within the specified range // library marker kkossev.deviceProfileLib, line 429 if ((dpMap.min != null && value < dpMap.min) || (dpMap.max != null && value > dpMap.max)) { // library marker kkossev.deviceProfileLib, line 430 logWarn "${device.displayName} validateAndScaleParameterValue: invalid ${dpMap.name} parameter value ${value} (scaled ${scaledValue}). Value must be within ${dpMap.min} and ${dpMap.max}" // library marker kkossev.deviceProfileLib, line 431 return null // library marker kkossev.deviceProfileLib, line 432 } // library marker kkossev.deviceProfileLib, line 433 //logTrace "validateAndScaleParameterValue returning scaledValue=${scaledValue}" // library marker kkossev.deviceProfileLib, line 434 return scaledValue // library marker kkossev.deviceProfileLib, line 435 } // library marker kkossev.deviceProfileLib, line 436 /** // library marker kkossev.deviceProfileLib, line 438 * Sets the value of a parameter for a device. // library marker kkossev.deviceProfileLib, line 439 * // library marker kkossev.deviceProfileLib, line 440 * @param par The parameter name. // library marker kkossev.deviceProfileLib, line 441 * @param val The parameter value. // library marker kkossev.deviceProfileLib, line 442 * @return true if the parameter was successfully set, false otherwise. // library marker kkossev.deviceProfileLib, line 443 */ // library marker kkossev.deviceProfileLib, line 444 public boolean setPar(final String parPar=null, final String val=null ) { // library marker kkossev.deviceProfileLib, line 445 List cmds = [] // library marker kkossev.deviceProfileLib, line 446 //Boolean validated = false // library marker kkossev.deviceProfileLib, line 447 logDebug "setPar(${parPar}, ${val})" // library marker kkossev.deviceProfileLib, line 448 if (DEVICE?.preferences == null || DEVICE?.preferences == [:]) { return false } // library marker kkossev.deviceProfileLib, line 449 if (parPar == null /*|| !(par in getValidParsPerModel())*/) { logInfo "setPar: 'parameter' must be one of these : ${getValidParsPerModel()}"; return false } // library marker kkossev.deviceProfileLib, line 450 String par = parPar.trim() // library marker kkossev.deviceProfileLib, line 451 Map dpMap = getPreferencesMapByName(par, false) // get the map for the parameter // library marker kkossev.deviceProfileLib, line 452 if ( dpMap == null || dpMap == [:]) { logInfo "setPar: tuyaDPs map not found for parameter ${par}"; return false } // library marker kkossev.deviceProfileLib, line 453 if (val == null) { logInfo "setPar: 'value' must be specified for parameter ${par} in the range ${dpMap.min} to ${dpMap.max}"; return false } // library marker kkossev.deviceProfileLib, line 454 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 455 def scaledValue = validateAndScaleParameterValue(dpMap, val as String) // convert the val to the correct type and scale it if needed // library marker kkossev.deviceProfileLib, line 456 if (scaledValue == null) { // library marker kkossev.deviceProfileLib, line 457 log.trace "$dpMap ${dpMap.map}" // library marker kkossev.deviceProfileLib, line 458 String helpTxt = "setPar: invalid parameter ${par} value ${val}." // library marker kkossev.deviceProfileLib, line 459 if (dpMap.min != null && dpMap.max != null) { helpTxt += " Must be in the range ${dpMap.min} to ${dpMap.max}" } // library marker kkossev.deviceProfileLib, line 460 if (dpMap.map != null) { helpTxt += " Must be one of ${dpMap.map}" } // library marker kkossev.deviceProfileLib, line 461 logInfo helpTxt // library marker kkossev.deviceProfileLib, line 462 return false // library marker kkossev.deviceProfileLib, line 463 } // library marker kkossev.deviceProfileLib, line 464 // if there is a dedicated set function, use it // library marker kkossev.deviceProfileLib, line 466 String capitalizedFirstChar = par[0].toUpperCase() + par[1..-1] // library marker kkossev.deviceProfileLib, line 467 String customSetFunction = "customSet${capitalizedFirstChar}" // library marker kkossev.deviceProfileLib, line 468 if (this.respondsTo(customSetFunction)) { // library marker kkossev.deviceProfileLib, line 469 logDebug "setPar: found customSetFunction=${setFunction}, scaledValue=${scaledValue} (val=${val})" // library marker kkossev.deviceProfileLib, line 470 // execute the customSetFunction // library marker kkossev.deviceProfileLib, line 471 try { cmds = "$customSetFunction"(scaledValue) } // library marker kkossev.deviceProfileLib, line 472 catch (e) { logWarn "setPar: Exception caught while processing $customSetFunction($scaledValue) (val=${val})) : '${e}'" ; return false } // library marker kkossev.deviceProfileLib, line 473 logDebug "customSetFunction result is ${cmds}" // library marker kkossev.deviceProfileLib, line 474 if (cmds != null && cmds != []) { // library marker kkossev.deviceProfileLib, line 475 logInfo "setPar: (1) successfluly executed setPar $customSetFunction($scaledValue)" // library marker kkossev.deviceProfileLib, line 476 sendZigbeeCommands( cmds ) // library marker kkossev.deviceProfileLib, line 477 return true // library marker kkossev.deviceProfileLib, line 478 } // library marker kkossev.deviceProfileLib, line 479 else { // library marker kkossev.deviceProfileLib, line 480 logWarn "setPar: customSetFunction $customSetFunction($scaledValue) returned null or empty list" // library marker kkossev.deviceProfileLib, line 481 // continue with the default processing // library marker kkossev.deviceProfileLib, line 482 } // library marker kkossev.deviceProfileLib, line 483 } // library marker kkossev.deviceProfileLib, line 484 if (isVirtual()) { // library marker kkossev.deviceProfileLib, line 485 // set a virtual attribute // library marker kkossev.deviceProfileLib, line 486 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 487 def valMiscType // library marker kkossev.deviceProfileLib, line 488 logDebug "setPar: found virtual attribute ${par} value ${val}" // library marker kkossev.deviceProfileLib, line 489 if (dpMap.type == 'enum') { // library marker kkossev.deviceProfileLib, line 490 // find the key for the value // library marker kkossev.deviceProfileLib, line 491 String key = dpMap.map.find { it.value == val }?.key // library marker kkossev.deviceProfileLib, line 492 logTrace "setPar: enum parameter ${val}. key=${key}" // library marker kkossev.deviceProfileLib, line 493 if (key == null) { // library marker kkossev.deviceProfileLib, line 494 logInfo "setPar: invalid virtual enum parameter ${val}. value must be one of ${dpMap.map}" // library marker kkossev.deviceProfileLib, line 495 return false // library marker kkossev.deviceProfileLib, line 496 } // library marker kkossev.deviceProfileLib, line 497 valMiscType = dpMap.map[key as int] // library marker kkossev.deviceProfileLib, line 498 logTrace "setPar: enum parameter ${val}. key=${key} valMiscType=${valMiscType} dpMap.map=${dpMap.map}" // library marker kkossev.deviceProfileLib, line 499 device.updateSetting("$par", [value:key as String, type:dpMap.type]) // library marker kkossev.deviceProfileLib, line 500 } // library marker kkossev.deviceProfileLib, line 501 else { // library marker kkossev.deviceProfileLib, line 502 valMiscType = val // library marker kkossev.deviceProfileLib, line 503 device.updateSetting("$par", [value:valMiscType, type:dpMap.type]) // library marker kkossev.deviceProfileLib, line 504 } // library marker kkossev.deviceProfileLib, line 505 String descriptionText = "${par} set to ${valMiscType}${dpMap.unit ?: ''} [virtual]" // library marker kkossev.deviceProfileLib, line 506 sendEvent(name:par, value:valMiscType, unit:dpMap.unit ?: '', isDigital: true) // library marker kkossev.deviceProfileLib, line 507 logInfo descriptionText // library marker kkossev.deviceProfileLib, line 508 return true // library marker kkossev.deviceProfileLib, line 509 } // library marker kkossev.deviceProfileLib, line 510 // check whether this is a tuya DP or a cluster:attribute parameter // library marker kkossev.deviceProfileLib, line 512 boolean isTuyaDP // library marker kkossev.deviceProfileLib, line 513 /* groovylint-disable-next-line Instanceof */ // library marker kkossev.deviceProfileLib, line 515 try { isTuyaDP = dpMap.dp instanceof Number } // library marker kkossev.deviceProfileLib, line 516 catch (e) { logWarn"setPar: (1) exception ${e} caught while checking isNumber() preference ${preference}" ; isTuyaDP = false } // library marker kkossev.deviceProfileLib, line 517 if (dpMap.dp != null && isTuyaDP) { // library marker kkossev.deviceProfileLib, line 518 // Tuya DP // library marker kkossev.deviceProfileLib, line 519 cmds = sendTuyaParameter(dpMap, par, scaledValue) // library marker kkossev.deviceProfileLib, line 520 if (cmds == null || cmds == []) { // library marker kkossev.deviceProfileLib, line 521 logWarn "setPar: sendTuyaParameter par ${par} scaledValue ${scaledValue} returned null or empty list" // library marker kkossev.deviceProfileLib, line 522 return false // library marker kkossev.deviceProfileLib, line 523 } // library marker kkossev.deviceProfileLib, line 524 else { // library marker kkossev.deviceProfileLib, line 525 logInfo "setPar: (2) sending parameter $par ($val (scaledValue=${scaledValue}))" // library marker kkossev.deviceProfileLib, line 526 sendZigbeeCommands(cmds) // library marker kkossev.deviceProfileLib, line 527 return false // library marker kkossev.deviceProfileLib, line 528 } // library marker kkossev.deviceProfileLib, line 529 } // library marker kkossev.deviceProfileLib, line 530 else if (dpMap.at != null) { // library marker kkossev.deviceProfileLib, line 531 // cluster:attribute // library marker kkossev.deviceProfileLib, line 532 logDebug "setPar: found at=${dpMap.at} dt=${dpMap.dt} mfgCode=${dpMap.mfgCode} scaledValue=${scaledValue} (val=${val})" // library marker kkossev.deviceProfileLib, line 533 int signedIntScaled = convertSignedInts(scaledValue, dpMap) // library marker kkossev.deviceProfileLib, line 534 cmds = zclWriteAttribute(dpMap, signedIntScaled) // library marker kkossev.deviceProfileLib, line 535 if (cmds == null || cmds == []) { // library marker kkossev.deviceProfileLib, line 536 logWarn "setPar: failed to write cluster:attribute ${dpMap.at} value ${scaledValue}" // library marker kkossev.deviceProfileLib, line 537 return false // library marker kkossev.deviceProfileLib, line 538 } // library marker kkossev.deviceProfileLib, line 539 } // library marker kkossev.deviceProfileLib, line 540 else { logWarn "setPar: invalid dp or at value ${dpMap.dp} for parameter ${par}" ; return false } // library marker kkossev.deviceProfileLib, line 541 logInfo "setPar: (3) successfluly executed setPar $customSetFunction($scaledValue)" // library marker kkossev.deviceProfileLib, line 542 sendZigbeeCommands( cmds ) // library marker kkossev.deviceProfileLib, line 543 return true // library marker kkossev.deviceProfileLib, line 544 } // library marker kkossev.deviceProfileLib, line 545 // function to send a Tuya command to data point taken from dpMap with value tuyaValue and type taken from dpMap // library marker kkossev.deviceProfileLib, line 547 // TODO - reuse it !!! // library marker kkossev.deviceProfileLib, line 548 /* groovylint-disable-next-line MethodParameterTypeRequired, NoDef */ // library marker kkossev.deviceProfileLib, line 549 public List sendTuyaParameter( Map dpMap, String par, tuyaValue) { // library marker kkossev.deviceProfileLib, line 550 //logDebug "sendTuyaParameter: trying to send parameter ${par} value ${tuyaValue}" // library marker kkossev.deviceProfileLib, line 551 List cmds = [] // library marker kkossev.deviceProfileLib, line 552 if (dpMap == null) { logWarn "sendTuyaParameter: tuyaDPs map not found for parameter ${par}" ; return [] } // library marker kkossev.deviceProfileLib, line 553 String dp = zigbee.convertToHexString(dpMap.dp, 2) // library marker kkossev.deviceProfileLib, line 554 if (dpMap.dp <= 0 || dpMap.dp >= 256) { // library marker kkossev.deviceProfileLib, line 555 logWarn "sendTuyaParameter: invalid dp ${dpMap.dp} for parameter ${par}" // library marker kkossev.deviceProfileLib, line 556 return [] // library marker kkossev.deviceProfileLib, line 557 } // library marker kkossev.deviceProfileLib, line 558 String dpType // library marker kkossev.deviceProfileLib, line 559 if (dpMap.dt == null) { // library marker kkossev.deviceProfileLib, line 560 dpType = dpMap.type == 'bool' ? DP_TYPE_BOOL : dpMap.type == 'enum' ? DP_TYPE_ENUM : (dpMap.type in ['value', 'number', 'decimal']) ? DP_TYPE_VALUE : null // library marker kkossev.deviceProfileLib, line 561 } // library marker kkossev.deviceProfileLib, line 562 else { // library marker kkossev.deviceProfileLib, line 563 dpType = dpMap.dt // "01" - bool, "02" - enum, "03" - value // library marker kkossev.deviceProfileLib, line 564 } // library marker kkossev.deviceProfileLib, line 565 if (dpType == null) { // library marker kkossev.deviceProfileLib, line 566 logWarn "sendTuyaParameter: invalid dpType ${dpMap.type} for parameter ${par}" // library marker kkossev.deviceProfileLib, line 567 return [] // library marker kkossev.deviceProfileLib, line 568 } // library marker kkossev.deviceProfileLib, line 569 // sendTuyaCommand // library marker kkossev.deviceProfileLib, line 570 String dpValHex = dpType == DP_TYPE_VALUE ? zigbee.convertToHexString(tuyaValue as int, 8) : zigbee.convertToHexString(tuyaValue as int, 2) // library marker kkossev.deviceProfileLib, line 571 logDebug "sendTuyaParameter: sending parameter ${par} dpValHex ${dpValHex} (raw=${tuyaValue}) Tuya dp=${dp} dpType=${dpType} " // library marker kkossev.deviceProfileLib, line 572 if (dpMap.tuyaCmd != null ) { // library marker kkossev.deviceProfileLib, line 573 cmds = sendTuyaCommand( dp, dpType, dpValHex, dpMap.tuyaCmd as int) // library marker kkossev.deviceProfileLib, line 574 } // library marker kkossev.deviceProfileLib, line 575 else { // library marker kkossev.deviceProfileLib, line 576 cmds = sendTuyaCommand( dp, dpType, dpValHex) // library marker kkossev.deviceProfileLib, line 577 } // library marker kkossev.deviceProfileLib, line 578 return cmds // library marker kkossev.deviceProfileLib, line 579 } // library marker kkossev.deviceProfileLib, line 580 private int convertSignedInts(int val, Map dpMap) { // library marker kkossev.deviceProfileLib, line 582 if (dpMap.dt == '0x28') { // library marker kkossev.deviceProfileLib, line 583 if (val > 127) { return (val as int) - 256 } // library marker kkossev.deviceProfileLib, line 584 else { return (val as int) } // library marker kkossev.deviceProfileLib, line 585 } // library marker kkossev.deviceProfileLib, line 586 else if (dpMap.dt == '0x29') { // library marker kkossev.deviceProfileLib, line 587 if (val > 32767) { return (val as int) - 65536 } // library marker kkossev.deviceProfileLib, line 588 else { return (val as int) } // library marker kkossev.deviceProfileLib, line 589 } // library marker kkossev.deviceProfileLib, line 590 else { return (val as int) } // library marker kkossev.deviceProfileLib, line 591 } // library marker kkossev.deviceProfileLib, line 592 /* groovylint-disable-next-line MethodParameterTypeRequired, NoDef */ // library marker kkossev.deviceProfileLib, line 594 public boolean sendAttribute(String par=null, val=null ) { // library marker kkossev.deviceProfileLib, line 595 List cmds = [] // library marker kkossev.deviceProfileLib, line 596 //Boolean validated = false // library marker kkossev.deviceProfileLib, line 597 logDebug "sendAttribute(${par}, ${val})" // library marker kkossev.deviceProfileLib, line 598 if (par == null || DEVICE?.preferences == null || DEVICE?.preferences == [:]) { logDebug 'DEVICE.preferences is empty!' ; return false } // library marker kkossev.deviceProfileLib, line 599 Map dpMap = getAttributesMap(par, false) // get the map for the attribute // library marker kkossev.deviceProfileLib, line 601 l//log.trace "sendAttribute: dpMap=${dpMap}" // library marker kkossev.deviceProfileLib, line 602 if (dpMap == null || dpMap?.isEmpty()) { logWarn "sendAttribute: map not found for parameter ${par}"; return false } // library marker kkossev.deviceProfileLib, line 603 if (val == null) { logWarn "sendAttribute: 'value' must be specified for parameter ${par} in the range ${dpMap.min} to ${dpMap.max}"; return false } // library marker kkossev.deviceProfileLib, line 604 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 605 def scaledValue = validateAndScaleParameterValue(dpMap, val as String) // convert the val to the correct type and scale it if needed // library marker kkossev.deviceProfileLib, line 606 if (scaledValue == null) { logWarn "sendAttribute: invalid parameter value ${val}. Must be in the range ${dpMap.min} to ${dpMap.max}"; return false } // library marker kkossev.deviceProfileLib, line 607 logDebug "sendAttribute: parameter ${par} value ${val}, type ${dpMap.type} validated and scaled to ${scaledValue} type=${dpMap.type}" // library marker kkossev.deviceProfileLib, line 608 // if there is a dedicated set function, use it // library marker kkossev.deviceProfileLib, line 609 String capitalizedFirstChar = par[0].toUpperCase() + par[1..-1] // library marker kkossev.deviceProfileLib, line 610 String customSetFunction = "customSet${capitalizedFirstChar}" // library marker kkossev.deviceProfileLib, line 611 if (this.respondsTo(customSetFunction) /*&& !(customSetFunction in ["setHeatingSetpoint", "setCoolingSetpoint", "setThermostatMode"])*/) { // library marker kkossev.deviceProfileLib, line 612 logDebug "sendAttribute: found customSetFunction=${customSetFunction}, scaledValue=${scaledValue} (val=${val})" // library marker kkossev.deviceProfileLib, line 613 // execute the customSetFunction // library marker kkossev.deviceProfileLib, line 614 try { // library marker kkossev.deviceProfileLib, line 615 cmds = "$customSetFunction"(scaledValue) // library marker kkossev.deviceProfileLib, line 616 } // library marker kkossev.deviceProfileLib, line 617 catch (e) { // library marker kkossev.deviceProfileLib, line 618 logWarn "sendAttribute: Exception '${e}'caught while processing $customSetFunction($scaledValue) (val=${val}))" // library marker kkossev.deviceProfileLib, line 619 return false // library marker kkossev.deviceProfileLib, line 620 } // library marker kkossev.deviceProfileLib, line 621 logDebug "customSetFunction result is ${cmds}" // library marker kkossev.deviceProfileLib, line 622 if (cmds != null && cmds != []) { // library marker kkossev.deviceProfileLib, line 623 logDebug "sendAttribute: successfluly executed sendAttribute $customSetFunction($scaledValue)" // library marker kkossev.deviceProfileLib, line 624 sendZigbeeCommands( cmds ) // library marker kkossev.deviceProfileLib, line 625 return true // library marker kkossev.deviceProfileLib, line 626 } // library marker kkossev.deviceProfileLib, line 627 else { // library marker kkossev.deviceProfileLib, line 628 logWarn "sendAttribute: customSetFunction $customSetFunction($scaledValue) returned null or empty list, continue with the default processing" // library marker kkossev.deviceProfileLib, line 629 // continue with the default processing // library marker kkossev.deviceProfileLib, line 630 } // library marker kkossev.deviceProfileLib, line 631 } // library marker kkossev.deviceProfileLib, line 632 else { // library marker kkossev.deviceProfileLib, line 633 logDebug "sendAttribute: SKIPPED customSetFunction ${customSetFunction}, continue with the default processing" // library marker kkossev.deviceProfileLib, line 634 } // library marker kkossev.deviceProfileLib, line 635 // check whether this is a tuya DP or a cluster:attribute parameter or a virtual device // library marker kkossev.deviceProfileLib, line 636 if (isVirtual()) { // library marker kkossev.deviceProfileLib, line 637 // send a virtual attribute // library marker kkossev.deviceProfileLib, line 638 logDebug "sendAttribute: found virtual attribute ${par} value ${val}" // library marker kkossev.deviceProfileLib, line 639 // patch !! // library marker kkossev.deviceProfileLib, line 640 if (par == 'heatingSetpoint') { // library marker kkossev.deviceProfileLib, line 641 sendHeatingSetpointEvent(val) // library marker kkossev.deviceProfileLib, line 642 } // library marker kkossev.deviceProfileLib, line 643 else { // library marker kkossev.deviceProfileLib, line 644 String descriptionText = "${par} is ${val} [virtual]" // library marker kkossev.deviceProfileLib, line 645 sendEvent(name:par, value:val, isDigital: true) // library marker kkossev.deviceProfileLib, line 646 logInfo descriptionText // library marker kkossev.deviceProfileLib, line 647 } // library marker kkossev.deviceProfileLib, line 648 return true // library marker kkossev.deviceProfileLib, line 649 } // library marker kkossev.deviceProfileLib, line 650 else { // library marker kkossev.deviceProfileLib, line 651 logDebug "sendAttribute: not a virtual device (device.controllerType = ${device.controllerType}), continue " // library marker kkossev.deviceProfileLib, line 652 } // library marker kkossev.deviceProfileLib, line 653 boolean isTuyaDP // library marker kkossev.deviceProfileLib, line 654 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 655 def preference = dpMap.dp // TODO - remove it? // library marker kkossev.deviceProfileLib, line 656 try { // library marker kkossev.deviceProfileLib, line 657 isTuyaDP = dpMap.dp instanceof Number // check if dpMap.dp is a number // library marker kkossev.deviceProfileLib, line 658 } // library marker kkossev.deviceProfileLib, line 659 catch (e) { // library marker kkossev.deviceProfileLib, line 660 if (debug) { log.warn "sendAttribute: exception ${e} caught while checking isNumber() preference ${preference}" } // library marker kkossev.deviceProfileLib, line 661 return false // library marker kkossev.deviceProfileLib, line 662 } // library marker kkossev.deviceProfileLib, line 663 if (dpMap.dp != null && isTuyaDP) { // library marker kkossev.deviceProfileLib, line 664 // Tuya DP // library marker kkossev.deviceProfileLib, line 665 cmds = sendTuyaParameter(dpMap, par, scaledValue) // library marker kkossev.deviceProfileLib, line 666 if (cmds == null || cmds == []) { // library marker kkossev.deviceProfileLib, line 667 logWarn "sendAttribute: sendTuyaParameter par ${par} scaledValue ${scaledValue} returned null or empty list" // library marker kkossev.deviceProfileLib, line 668 return false // library marker kkossev.deviceProfileLib, line 669 } // library marker kkossev.deviceProfileLib, line 670 else { // library marker kkossev.deviceProfileLib, line 671 logDebug "sendAttribute: successfluly executed sendAttribute $customSetFunction($val (scaledValue=${scaledValue}))" // library marker kkossev.deviceProfileLib, line 672 sendZigbeeCommands( cmds ) // library marker kkossev.deviceProfileLib, line 673 return true // library marker kkossev.deviceProfileLib, line 674 } // library marker kkossev.deviceProfileLib, line 675 } // library marker kkossev.deviceProfileLib, line 676 /* groovylint-disable-next-line EmptyIfStatement */ // library marker kkossev.deviceProfileLib, line 677 else if (dpMap.at != null && dpMap.at == 'virtual') { // library marker kkossev.deviceProfileLib, line 678 // send a virtual attribute // library marker kkossev.deviceProfileLib, line 679 } // library marker kkossev.deviceProfileLib, line 680 else if (dpMap.at != null) { // library marker kkossev.deviceProfileLib, line 681 // cluster:attribute // library marker kkossev.deviceProfileLib, line 682 cmds = zclWriteAttribute(dpMap, scaledValue) // library marker kkossev.deviceProfileLib, line 683 if (cmds == null || cmds == []) { // library marker kkossev.deviceProfileLib, line 684 logWarn "sendAttribute: failed to write cluster:attribute ${dpMap.at} value ${scaledValue}" // library marker kkossev.deviceProfileLib, line 685 return false // library marker kkossev.deviceProfileLib, line 686 } // library marker kkossev.deviceProfileLib, line 687 } // library marker kkossev.deviceProfileLib, line 688 else { // library marker kkossev.deviceProfileLib, line 689 logWarn "sendAttribute: invalid dp or at value ${dpMap.dp} for parameter ${par}" // library marker kkossev.deviceProfileLib, line 690 return false // library marker kkossev.deviceProfileLib, line 691 } // library marker kkossev.deviceProfileLib, line 692 logDebug "sendAttribute: successfluly executed sendAttribute $customSetFunction($scaledValue)" // library marker kkossev.deviceProfileLib, line 693 sendZigbeeCommands( cmds ) // library marker kkossev.deviceProfileLib, line 694 return true // library marker kkossev.deviceProfileLib, line 695 } // library marker kkossev.deviceProfileLib, line 696 /** // library marker kkossev.deviceProfileLib, line 698 * SENDS a list of Zigbee commands to be sent to the device. // library marker kkossev.deviceProfileLib, line 699 * @param command - The command to send. Must be one of the commands defined in the DEVICE.commands map. // library marker kkossev.deviceProfileLib, line 700 * @param val - The value to send with the command, can be null. // library marker kkossev.deviceProfileLib, line 701 * @return true on success, false otherwise. // library marker kkossev.deviceProfileLib, line 702 */ // library marker kkossev.deviceProfileLib, line 703 public boolean sendCommand(final String command_orig=null, final String val_orig=null) { // library marker kkossev.deviceProfileLib, line 704 //logDebug "sending command ${command}(${val}))" // library marker kkossev.deviceProfileLib, line 705 final String command = command_orig?.trim() // library marker kkossev.deviceProfileLib, line 706 final String val = val_orig?.trim() // library marker kkossev.deviceProfileLib, line 707 List cmds = [] // library marker kkossev.deviceProfileLib, line 708 Map supportedCommandsMap = DEVICE?.commands as Map // library marker kkossev.deviceProfileLib, line 709 if (supportedCommandsMap == null || supportedCommandsMap?.isEmpty()) { // library marker kkossev.deviceProfileLib, line 710 logInfo "sendCommand: no commands defined for device profile ${getDeviceProfile()} !" // library marker kkossev.deviceProfileLib, line 711 return false // library marker kkossev.deviceProfileLib, line 712 } // library marker kkossev.deviceProfileLib, line 713 // TODO: compare ignoring the upper/lower case of the command. // library marker kkossev.deviceProfileLib, line 714 List supportedCommandsList = DEVICE?.commands?.keySet() as List // library marker kkossev.deviceProfileLib, line 715 // check if the command is defined in the DEVICE commands map // library marker kkossev.deviceProfileLib, line 716 if (command == null || !(command in supportedCommandsList)) { // library marker kkossev.deviceProfileLib, line 717 logInfo "sendCommand: the command ${(command ?: '')} for device profile '${DEVICE?.description}' must be one of these : ${supportedCommandsList}" // library marker kkossev.deviceProfileLib, line 718 return false // library marker kkossev.deviceProfileLib, line 719 } // library marker kkossev.deviceProfileLib, line 720 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 721 def func, funcResult // library marker kkossev.deviceProfileLib, line 722 try { // library marker kkossev.deviceProfileLib, line 723 func = DEVICE?.commands.find { it.key == command }.value // library marker kkossev.deviceProfileLib, line 724 // added 01/25/2025 : the commands now can be shorted : instead of a map kay and value 'printFingerprints':'printFingerprints' we can skip the value when it is the same: 'printFingerprints:' - the value is the same as the key // library marker kkossev.deviceProfileLib, line 725 if (func == null || func == '') { // library marker kkossev.deviceProfileLib, line 726 func = command // library marker kkossev.deviceProfileLib, line 727 } // library marker kkossev.deviceProfileLib, line 728 if (val != null && val != '') { // library marker kkossev.deviceProfileLib, line 729 logInfo "executed $func($val)" // library marker kkossev.deviceProfileLib, line 730 funcResult = "${func}"(val) // library marker kkossev.deviceProfileLib, line 731 } // library marker kkossev.deviceProfileLib, line 732 else { // library marker kkossev.deviceProfileLib, line 733 logInfo "executed $func()" // library marker kkossev.deviceProfileLib, line 734 funcResult = "${func}"() // library marker kkossev.deviceProfileLib, line 735 } // library marker kkossev.deviceProfileLib, line 736 } // library marker kkossev.deviceProfileLib, line 737 catch (e) { // library marker kkossev.deviceProfileLib, line 738 logWarn "sendCommand: Exception '${e}' caught while processing $func(${val})" // library marker kkossev.deviceProfileLib, line 739 return false // library marker kkossev.deviceProfileLib, line 740 } // library marker kkossev.deviceProfileLib, line 741 // funcResult is expected to be list of commands to be sent to the device, but can also return boolean or null // library marker kkossev.deviceProfileLib, line 742 // check if the result is a list of commands // library marker kkossev.deviceProfileLib, line 743 /* groovylint-disable-next-line Instanceof */ // library marker kkossev.deviceProfileLib, line 744 if (funcResult instanceof List) { // library marker kkossev.deviceProfileLib, line 745 cmds = funcResult // library marker kkossev.deviceProfileLib, line 746 if (cmds != null && cmds != []) { // library marker kkossev.deviceProfileLib, line 747 sendZigbeeCommands( cmds ) // library marker kkossev.deviceProfileLib, line 748 } // library marker kkossev.deviceProfileLib, line 749 } // library marker kkossev.deviceProfileLib, line 750 else if (funcResult == null) { // library marker kkossev.deviceProfileLib, line 751 return false // library marker kkossev.deviceProfileLib, line 752 } // library marker kkossev.deviceProfileLib, line 753 else { // library marker kkossev.deviceProfileLib, line 754 logDebug "sendCommand: $func(${val}) returned ${funcResult} instead of a list of commands!" // library marker kkossev.deviceProfileLib, line 755 return false // library marker kkossev.deviceProfileLib, line 756 } // library marker kkossev.deviceProfileLib, line 757 return true // library marker kkossev.deviceProfileLib, line 758 } // library marker kkossev.deviceProfileLib, line 759 /** // library marker kkossev.deviceProfileLib, line 761 * This method takes a string parameter and a boolean debug flag as input and returns a map containing the input details. // library marker kkossev.deviceProfileLib, line 762 * The method checks if the input parameter is defined in the device preferences and returns null if it is not. // library marker kkossev.deviceProfileLib, line 763 * It then checks if the input parameter is a boolean value and skips it if it is. // library marker kkossev.deviceProfileLib, line 764 * The method also checks if the input parameter is a number and sets the isTuyaDP flag accordingly. // library marker kkossev.deviceProfileLib, line 765 * If the input parameter is read-only, the method returns null. // library marker kkossev.deviceProfileLib, line 766 * The method then populates the input map with the name, type, title, description, range, options, and default value of the input parameter. // library marker kkossev.deviceProfileLib, line 767 * If the input parameter type is not supported, the method returns null. // library marker kkossev.deviceProfileLib, line 768 * @param param The input parameter to be checked. // library marker kkossev.deviceProfileLib, line 769 * @param debug A boolean flag indicating whether to log debug messages or not. // library marker kkossev.deviceProfileLib, line 770 * @return A map containing the input details. // library marker kkossev.deviceProfileLib, line 771 */ // library marker kkossev.deviceProfileLib, line 772 public Map inputIt(String paramPar, boolean debug = false) { // library marker kkossev.deviceProfileLib, line 773 String param = paramPar.trim() // library marker kkossev.deviceProfileLib, line 774 Map input = [:] // library marker kkossev.deviceProfileLib, line 775 Map foundMap = [:] // library marker kkossev.deviceProfileLib, line 776 if (!(param in DEVICE?.preferences)) { if (debug) { log.warn "inputIt: preference ${param} not defined for this device!" } ; return [:] } // library marker kkossev.deviceProfileLib, line 777 Object preference // library marker kkossev.deviceProfileLib, line 778 try { preference = DEVICE?.preferences["$param"] } // library marker kkossev.deviceProfileLib, line 779 catch (e) { if (debug) { log.warn "inputIt: exception ${e} caught while parsing preference ${param} value ${preference}" } ; return [:] } // library marker kkossev.deviceProfileLib, line 780 // check for boolean values // library marker kkossev.deviceProfileLib, line 781 try { if (preference in [true, false]) { if (debug) { log.warn "inputIt: preference ${param} is boolean value ${preference} - skipping it for now!" } ; return [:] } } // library marker kkossev.deviceProfileLib, line 782 catch (e) { if (debug) { log.warn "inputIt: exception ${e} caught while checking for boolean values preference ${param} value ${preference}" } ; return [:] } // library marker kkossev.deviceProfileLib, line 783 /* // library marker kkossev.deviceProfileLib, line 784 // TODO - check if this is neccessary? isTuyaDP is not defined! // library marker kkossev.deviceProfileLib, line 785 try { isTuyaDP = preference.isNumber() } // library marker kkossev.deviceProfileLib, line 786 catch (e) { if (debug) { log.warn "inputIt: exception ${e} caught while checking isNumber() preference ${param} value ${preference}" } ; return [:] } // library marker kkossev.deviceProfileLib, line 787 */ // library marker kkossev.deviceProfileLib, line 788 //if (debug) log.debug "inputIt: preference ${param} found. value is ${preference} isTuyaDP=${isTuyaDP}" // library marker kkossev.deviceProfileLib, line 789 foundMap = getPreferencesMapByName(param) // library marker kkossev.deviceProfileLib, line 790 //if (debug) log.debug "foundMap = ${foundMap}" // library marker kkossev.deviceProfileLib, line 791 if (foundMap == null || foundMap?.isEmpty()) { if (debug) { log.warn "inputIt: map not found for param '${param}'!" } ; return [:] } // library marker kkossev.deviceProfileLib, line 792 if (foundMap.rw != 'rw') { if (debug) { log.warn "inputIt: param '${param}' is read only!" } ; return [:] } // library marker kkossev.deviceProfileLib, line 793 if (foundMap.advanced != null && foundMap.advanced == true && settings.advancedOptions != true) { // library marker kkossev.deviceProfileLib, line 794 if (debug) { log.debug "inputIt: param '${param}' is advanced!" } // library marker kkossev.deviceProfileLib, line 795 return [:] // library marker kkossev.deviceProfileLib, line 796 } // library marker kkossev.deviceProfileLib, line 797 input.name = foundMap.name // library marker kkossev.deviceProfileLib, line 798 input.type = foundMap.type // bool, enum, number, decimal // library marker kkossev.deviceProfileLib, line 799 input.title = foundMap.title // library marker kkossev.deviceProfileLib, line 800 //input.description = (foundMap.description ?: foundMap.title)?.replaceAll(/<\/?b>/, '') // if description is not defined, use the title // library marker kkossev.deviceProfileLib, line 801 input.description = foundMap.description ?: '' // if description is not defined, skip it // library marker kkossev.deviceProfileLib, line 802 if (input.type in ['number', 'decimal']) { // library marker kkossev.deviceProfileLib, line 803 if (foundMap.min != null && foundMap.max != null) { // library marker kkossev.deviceProfileLib, line 804 //input.range = "${foundMap.min}..${foundMap.max}" // library marker kkossev.deviceProfileLib, line 805 input.range = "${Math.ceil(foundMap.min) as int}..${Math.ceil(foundMap.max) as int}" // library marker kkossev.deviceProfileLib, line 806 } // library marker kkossev.deviceProfileLib, line 807 if (input.range != null && input.description != null) { // library marker kkossev.deviceProfileLib, line 808 if (input.description != '') { input.description += '
' } // library marker kkossev.deviceProfileLib, line 809 input.description += "Range: ${input.range}" // library marker kkossev.deviceProfileLib, line 810 if (foundMap.unit != null && foundMap.unit != '') { // library marker kkossev.deviceProfileLib, line 811 input.description += " (${foundMap.unit})" // library marker kkossev.deviceProfileLib, line 812 } // library marker kkossev.deviceProfileLib, line 813 } // library marker kkossev.deviceProfileLib, line 814 } // library marker kkossev.deviceProfileLib, line 815 /* groovylint-disable-next-line SpaceAfterClosingBrace */ // library marker kkossev.deviceProfileLib, line 816 else if (input.type == 'enum') { // library marker kkossev.deviceProfileLib, line 817 input.options = foundMap.map // library marker kkossev.deviceProfileLib, line 818 }/* // library marker kkossev.deviceProfileLib, line 819 else if (input.type == "bool") { // library marker kkossev.deviceProfileLib, line 820 input.options = ["true", "false"] // library marker kkossev.deviceProfileLib, line 821 }*/ // library marker kkossev.deviceProfileLib, line 822 else { // library marker kkossev.deviceProfileLib, line 823 if (debug) { log.warn "inputIt: unsupported type ${input.type} for param '${param}'!" } // library marker kkossev.deviceProfileLib, line 824 return [:] // library marker kkossev.deviceProfileLib, line 825 } // library marker kkossev.deviceProfileLib, line 826 if (input.defVal != null) { // library marker kkossev.deviceProfileLib, line 827 input.defVal = foundMap.defVal // library marker kkossev.deviceProfileLib, line 828 } // library marker kkossev.deviceProfileLib, line 829 return input // library marker kkossev.deviceProfileLib, line 830 } // library marker kkossev.deviceProfileLib, line 831 /** // library marker kkossev.deviceProfileLib, line 833 * Returns the device name and profile based on the device model and manufacturer. // library marker kkossev.deviceProfileLib, line 834 * @param model The device model (optional). If not provided, it will be retrieved from the device data value. // library marker kkossev.deviceProfileLib, line 835 * @param manufacturer The device manufacturer (optional). If not provided, it will be retrieved from the device data value. // library marker kkossev.deviceProfileLib, line 836 * @return A list containing the device name and profile. // library marker kkossev.deviceProfileLib, line 837 */ // library marker kkossev.deviceProfileLib, line 838 public List getDeviceNameAndProfile(String model=null, String manufacturer=null) { // library marker kkossev.deviceProfileLib, line 839 String deviceName = UNKNOWN, deviceProfile = UNKNOWN // library marker kkossev.deviceProfileLib, line 840 String deviceModel = model != null ? model : device.getDataValue('model') ?: UNKNOWN // library marker kkossev.deviceProfileLib, line 841 String deviceManufacturer = manufacturer != null ? manufacturer : device.getDataValue('manufacturer') ?: UNKNOWN // library marker kkossev.deviceProfileLib, line 842 deviceProfilesV3.each { profileName, profileMap -> // library marker kkossev.deviceProfileLib, line 843 profileMap.fingerprints.each { fingerprint -> // library marker kkossev.deviceProfileLib, line 844 if (fingerprint.model == deviceModel && fingerprint.manufacturer == deviceManufacturer) { // library marker kkossev.deviceProfileLib, line 845 deviceProfile = profileName // library marker kkossev.deviceProfileLib, line 846 deviceName = fingerprint.deviceJoinName ?: deviceProfilesV3[deviceProfile].description ?: UNKNOWN // library marker kkossev.deviceProfileLib, line 847 logDebug "found exact match for model ${deviceModel} manufacturer ${deviceManufacturer} : profileName=${deviceProfile} deviceName =${deviceName}" // library marker kkossev.deviceProfileLib, line 848 return [deviceName, deviceProfile] // library marker kkossev.deviceProfileLib, line 849 } // library marker kkossev.deviceProfileLib, line 850 } // library marker kkossev.deviceProfileLib, line 851 } // library marker kkossev.deviceProfileLib, line 852 if (deviceProfile == UNKNOWN) { // library marker kkossev.deviceProfileLib, line 853 logWarn "getDeviceNameAndProfile: NOT FOUND! deviceName =${deviceName} profileName=${deviceProfile} for model ${deviceModel} manufacturer ${deviceManufacturer}" // library marker kkossev.deviceProfileLib, line 854 } // library marker kkossev.deviceProfileLib, line 855 return [deviceName, deviceProfile] // library marker kkossev.deviceProfileLib, line 856 } // library marker kkossev.deviceProfileLib, line 857 // called from initializeVars( fullInit = true) // library marker kkossev.deviceProfileLib, line 859 public void setDeviceNameAndProfile(String model=null, String manufacturer=null) { // library marker kkossev.deviceProfileLib, line 860 def (String deviceName, String deviceProfile) = getDeviceNameAndProfile(model, manufacturer) // library marker kkossev.deviceProfileLib, line 861 if (deviceProfile == null || deviceProfile == UNKNOWN) { // library marker kkossev.deviceProfileLib, line 862 logInfo "unknown model ${deviceModel} manufacturer ${deviceManufacturer}" // library marker kkossev.deviceProfileLib, line 863 // don't change the device name when unknown // library marker kkossev.deviceProfileLib, line 864 state.deviceProfile = UNKNOWN // library marker kkossev.deviceProfileLib, line 865 } // library marker kkossev.deviceProfileLib, line 866 String dataValueModel = model != null ? model : device.getDataValue('model') ?: UNKNOWN // library marker kkossev.deviceProfileLib, line 867 String dataValueManufacturer = manufacturer != null ? manufacturer : device.getDataValue('manufacturer') ?: UNKNOWN // library marker kkossev.deviceProfileLib, line 868 if (deviceName != NULL && deviceName != UNKNOWN) { // library marker kkossev.deviceProfileLib, line 869 device.setName(deviceName) // library marker kkossev.deviceProfileLib, line 870 state.deviceProfile = deviceProfile // library marker kkossev.deviceProfileLib, line 871 device.updateSetting('forcedProfile', [value:deviceProfilesV3[deviceProfile]?.description, type:'enum']) // library marker kkossev.deviceProfileLib, line 872 logInfo "device model ${dataValueModel} manufacturer ${dataValueManufacturer} was set to : deviceProfile=${deviceProfile} : deviceName=${deviceName}" // library marker kkossev.deviceProfileLib, line 873 } else { // library marker kkossev.deviceProfileLib, line 874 logInfo "device model ${dataValueModel} manufacturer ${dataValueManufacturer} was not found!" // library marker kkossev.deviceProfileLib, line 875 } // library marker kkossev.deviceProfileLib, line 876 } // library marker kkossev.deviceProfileLib, line 877 public List refreshFromConfigureReadList(List refreshList) { // library marker kkossev.deviceProfileLib, line 879 logDebug "refreshFromConfigureReadList(${refreshList})" // library marker kkossev.deviceProfileLib, line 880 List cmds = [] // library marker kkossev.deviceProfileLib, line 881 if (refreshList != null && !refreshList.isEmpty()) { // library marker kkossev.deviceProfileLib, line 882 //List refreshList = DEVICE.refresh // library marker kkossev.deviceProfileLib, line 883 for (String k : refreshList) { // library marker kkossev.deviceProfileLib, line 884 k = k.replaceAll('\\[|\\]', '') // library marker kkossev.deviceProfileLib, line 885 if (k != null) { // library marker kkossev.deviceProfileLib, line 886 // check whether the string in the refreshList matches an attribute name in the DEVICE.attributes list // library marker kkossev.deviceProfileLib, line 887 Map map = DEVICE.attributes.find { it.name == k } // library marker kkossev.deviceProfileLib, line 888 if (map != null) { // library marker kkossev.deviceProfileLib, line 889 Map mfgCode = map.mfgCode != null ? ['mfgCode':map.mfgCode] : [:] // library marker kkossev.deviceProfileLib, line 890 cmds += zigbee.readAttribute(hubitat.helper.HexUtils.hexStringToInt((map.at).split(':')[0]), hubitat.helper.HexUtils.hexStringToInt((map.at).split(':')[1]), mfgCode, delay = 100) // library marker kkossev.deviceProfileLib, line 891 } // library marker kkossev.deviceProfileLib, line 892 // check whether the string in the refreshList matches a method defined somewhere in the code // library marker kkossev.deviceProfileLib, line 893 if (this.respondsTo(k)) { // library marker kkossev.deviceProfileLib, line 894 cmds += this."${k}"() // library marker kkossev.deviceProfileLib, line 895 } // library marker kkossev.deviceProfileLib, line 896 } // library marker kkossev.deviceProfileLib, line 897 } // library marker kkossev.deviceProfileLib, line 898 } // library marker kkossev.deviceProfileLib, line 899 return cmds // library marker kkossev.deviceProfileLib, line 900 } // library marker kkossev.deviceProfileLib, line 901 // called from customRefresh() in the device drivers // library marker kkossev.deviceProfileLib, line 903 public List refreshFromDeviceProfileList() { // library marker kkossev.deviceProfileLib, line 904 logDebug 'refreshFromDeviceProfileList()' // library marker kkossev.deviceProfileLib, line 905 List cmds = [] // library marker kkossev.deviceProfileLib, line 906 if (DEVICE?.refresh != null) { // library marker kkossev.deviceProfileLib, line 907 List refreshList = DEVICE.refresh // library marker kkossev.deviceProfileLib, line 908 for (String k : refreshList) { // library marker kkossev.deviceProfileLib, line 909 k = k.replaceAll('\\[|\\]', '') // library marker kkossev.deviceProfileLib, line 910 if (k != null) { // library marker kkossev.deviceProfileLib, line 911 // check whether the string in the refreshList matches an attribute name in the DEVICE.attributes list // library marker kkossev.deviceProfileLib, line 912 Map map = DEVICE.attributes.find { it.name == k } // library marker kkossev.deviceProfileLib, line 913 if (map != null) { // library marker kkossev.deviceProfileLib, line 914 Map mfgCode = map.mfgCode != null ? ['mfgCode':map.mfgCode] : [:] // library marker kkossev.deviceProfileLib, line 915 cmds += zigbee.readAttribute(hubitat.helper.HexUtils.hexStringToInt((map.at).split(':')[0]), hubitat.helper.HexUtils.hexStringToInt((map.at).split(':')[1]), mfgCode, delay = 100) // library marker kkossev.deviceProfileLib, line 916 } // library marker kkossev.deviceProfileLib, line 917 // check whether the string in the refreshList matches a method defined somewhere in the code // library marker kkossev.deviceProfileLib, line 918 if (this.respondsTo(k)) { // library marker kkossev.deviceProfileLib, line 919 cmds += this."${k}"() // library marker kkossev.deviceProfileLib, line 920 } // library marker kkossev.deviceProfileLib, line 921 } // library marker kkossev.deviceProfileLib, line 922 } // library marker kkossev.deviceProfileLib, line 923 } // library marker kkossev.deviceProfileLib, line 924 return cmds // library marker kkossev.deviceProfileLib, line 925 } // library marker kkossev.deviceProfileLib, line 926 // TODO! - remove? // library marker kkossev.deviceProfileLib, line 928 List refreshDeviceProfile() { // library marker kkossev.deviceProfileLib, line 929 List cmds = [] // library marker kkossev.deviceProfileLib, line 930 if (cmds == []) { cmds = ['delay 299'] } // library marker kkossev.deviceProfileLib, line 931 logDebug "refreshDeviceProfile() : ${cmds}" // library marker kkossev.deviceProfileLib, line 932 return cmds // library marker kkossev.deviceProfileLib, line 933 } // library marker kkossev.deviceProfileLib, line 934 // TODO ! - remove? // library marker kkossev.deviceProfileLib, line 936 List configureDeviceProfile() { // library marker kkossev.deviceProfileLib, line 937 List cmds = [] // library marker kkossev.deviceProfileLib, line 938 logDebug "configureDeviceProfile() : ${cmds}" // library marker kkossev.deviceProfileLib, line 939 if (cmds == []) { cmds = ['delay 299'] } // library marker kkossev.deviceProfileLib, line 940 return cmds // library marker kkossev.deviceProfileLib, line 941 } // library marker kkossev.deviceProfileLib, line 942 // TODO! - remove? // library marker kkossev.deviceProfileLib, line 944 List initializeDeviceProfile() { // library marker kkossev.deviceProfileLib, line 945 List cmds = [] // library marker kkossev.deviceProfileLib, line 946 logDebug "initializeDeviceProfile() : ${cmds}" // library marker kkossev.deviceProfileLib, line 947 if (cmds == []) { cmds = ['delay 299',] } // library marker kkossev.deviceProfileLib, line 948 return cmds // library marker kkossev.deviceProfileLib, line 949 } // library marker kkossev.deviceProfileLib, line 950 public void deviceProfileInitializeVars(boolean fullInit=false) { // library marker kkossev.deviceProfileLib, line 952 logDebug "deviceProfileInitializeVars(${fullInit})" // library marker kkossev.deviceProfileLib, line 953 if (state.deviceProfile == null) { // library marker kkossev.deviceProfileLib, line 954 setDeviceNameAndProfile() // library marker kkossev.deviceProfileLib, line 955 } // library marker kkossev.deviceProfileLib, line 956 } // library marker kkossev.deviceProfileLib, line 957 public void initEventsDeviceProfile(boolean fullInit=false) { // library marker kkossev.deviceProfileLib, line 959 String ps = DEVICE?.device?.powerSource // library marker kkossev.deviceProfileLib, line 960 logDebug "initEventsDeviceProfile(${fullInit}) for deviceProfile=${state.deviceProfile} DEVICE?.device?.powerSource=${ps} ps.isEmpty()=${ps?.isEmpty()}" // library marker kkossev.deviceProfileLib, line 961 if (ps != null && !ps.isEmpty()) { // library marker kkossev.deviceProfileLib, line 962 sendEvent(name: 'powerSource', value: ps, descriptionText: "Power Source set to '${ps}'", type: 'digital') // library marker kkossev.deviceProfileLib, line 963 } // library marker kkossev.deviceProfileLib, line 964 } // library marker kkossev.deviceProfileLib, line 965 ///////////////////////////// Tuya DPs ///////////////////////////////// // library marker kkossev.deviceProfileLib, line 967 // // library marker kkossev.deviceProfileLib, line 969 // called from parse() // library marker kkossev.deviceProfileLib, line 970 // returns: true - do not process this message if the spammy DP is defined in the spammyDPsToIgnore element of the active Device Profile // library marker kkossev.deviceProfileLib, line 971 // false - the processing can continue // library marker kkossev.deviceProfileLib, line 972 // // library marker kkossev.deviceProfileLib, line 973 public boolean isSpammyDPsToIgnore(Map descMap) { // library marker kkossev.deviceProfileLib, line 974 //log.trace "isSpammyDPsToIgnore: ${state.deviceProfile == 'TS0225_LINPTECH_RADAR'} ${descMap.cluster == 'E002'} ${descMap.attrId == 'E00A'} ${settings?.ignoreDistance == true}" // library marker kkossev.deviceProfileLib, line 975 if (state.deviceProfile == 'TS0225_LINPTECH_RADAR' && descMap.cluster == 'E002' && descMap.attrId == 'E00A' && settings?.ignoreDistance == true) { return true } // library marker kkossev.deviceProfileLib, line 976 if (!(descMap?.clusterId == 'EF00' && (descMap?.command in ['01', '02']))) { return false } // library marker kkossev.deviceProfileLib, line 977 if (descMap?.data?.size <= 2) { return false } // library marker kkossev.deviceProfileLib, line 978 int dp = zigbee.convertHexToInt(descMap.data[2]) // library marker kkossev.deviceProfileLib, line 979 List spammyList = deviceProfilesV3[getDeviceProfile()]?.spammyDPsToIgnore as List // library marker kkossev.deviceProfileLib, line 980 return (spammyList != null && (dp in spammyList) && ((settings?.ignoreDistance ?: false) == true)) // library marker kkossev.deviceProfileLib, line 981 } // library marker kkossev.deviceProfileLib, line 982 // // library marker kkossev.deviceProfileLib, line 984 // called from processTuyaDP(), processTuyaDPfromDeviceProfile(), isChattyDeviceReport() // library marker kkossev.deviceProfileLib, line 985 // returns: true - do not generate Debug log messages if the chatty DP is defined in the spammyDPsToNotTrace element of the active Device Profile // library marker kkossev.deviceProfileLib, line 986 // false - debug logs can be generated // library marker kkossev.deviceProfileLib, line 987 // // library marker kkossev.deviceProfileLib, line 988 public boolean isSpammyDPsToNotTrace(Map descMap) { // library marker kkossev.deviceProfileLib, line 989 //log.trace "isSpammyDPsToNotTrace: ${state.deviceProfile == 'TS0225_LINPTECH_RADAR'} ${descMap.cluster == 'E002'} ${descMap.attrId == 'E00A'} ${settings?.ignoreDistance == true}" // library marker kkossev.deviceProfileLib, line 990 if (state.deviceProfile == 'TS0225_LINPTECH_RADAR' && descMap.cluster == 'E002' && descMap.attrId == 'E00A' && settings?.ignoreDistance == true) { return true } // library marker kkossev.deviceProfileLib, line 991 if (!(descMap?.clusterId == 'EF00' && (descMap?.command in ['01', '02']))) { return false } // library marker kkossev.deviceProfileLib, line 992 if (descMap?.data?.size <= 2) { return false } // library marker kkossev.deviceProfileLib, line 993 int dp = zigbee.convertHexToInt(descMap.data[2]) // library marker kkossev.deviceProfileLib, line 994 List spammyList = deviceProfilesV3[getDeviceProfile()]?.spammyDPsToNotTrace as List // library marker kkossev.deviceProfileLib, line 995 return (spammyList != null && (dp in spammyList)) // library marker kkossev.deviceProfileLib, line 996 } // library marker kkossev.deviceProfileLib, line 997 // all DPs are spammy - sent periodically! (this function is not used?) // library marker kkossev.deviceProfileLib, line 999 public boolean isSpammyDeviceProfile() { // library marker kkossev.deviceProfileLib, line 1000 if (deviceProfilesV3 == null || deviceProfilesV3[getDeviceProfile()] == null) { return false } // library marker kkossev.deviceProfileLib, line 1001 Boolean isSpammy = deviceProfilesV3[getDeviceProfile()]?.device?.isSpammy ?: false // library marker kkossev.deviceProfileLib, line 1002 return isSpammy // library marker kkossev.deviceProfileLib, line 1003 } // library marker kkossev.deviceProfileLib, line 1004 /* groovylint-disable-next-line UnusedMethodParameter */ // library marker kkossev.deviceProfileLib, line 1006 private List compareAndConvertStrings(final Map foundItem, String tuyaValue, String hubitatValue) { // library marker kkossev.deviceProfileLib, line 1007 String convertedValue = tuyaValue // library marker kkossev.deviceProfileLib, line 1008 boolean isEqual = ((tuyaValue as String) == (hubitatValue as String)) // because the events(attributes) are always strings // library marker kkossev.deviceProfileLib, line 1009 if (foundItem?.scale != null || foundItem?.scale != 0 || foundItem?.scale != 1) { // library marker kkossev.deviceProfileLib, line 1010 logTrace "compareAndConvertStrings: scaling: foundItem.scale=${foundItem.scale} tuyaValue=${tuyaValue} hubitatValue=${hubitatValue}" // library marker kkossev.deviceProfileLib, line 1011 } // library marker kkossev.deviceProfileLib, line 1012 return [isEqual, convertedValue] // library marker kkossev.deviceProfileLib, line 1013 } // library marker kkossev.deviceProfileLib, line 1014 private List compareAndConvertNumbers(final Map foundItem, int tuyaValue, int hubitatValue) { // library marker kkossev.deviceProfileLib, line 1016 Integer convertedValue // library marker kkossev.deviceProfileLib, line 1017 boolean isEqual // library marker kkossev.deviceProfileLib, line 1018 if (foundItem?.scale == null || foundItem?.scale == 0 || foundItem?.scale == 1) { // compare as integer // library marker kkossev.deviceProfileLib, line 1019 convertedValue = tuyaValue as int // library marker kkossev.deviceProfileLib, line 1020 } // library marker kkossev.deviceProfileLib, line 1021 else { // library marker kkossev.deviceProfileLib, line 1022 convertedValue = ((tuyaValue as double) / (foundItem.scale as double)) as int // library marker kkossev.deviceProfileLib, line 1023 } // library marker kkossev.deviceProfileLib, line 1024 isEqual = ((convertedValue as int) == (hubitatValue as int)) // library marker kkossev.deviceProfileLib, line 1025 return [isEqual, convertedValue] // library marker kkossev.deviceProfileLib, line 1026 } // library marker kkossev.deviceProfileLib, line 1027 private List compareAndConvertDecimals(final Map foundItem, double tuyaValue, double hubitatValue) { // library marker kkossev.deviceProfileLib, line 1029 Double convertedValue // library marker kkossev.deviceProfileLib, line 1030 if (foundItem?.scale == null || foundItem?.scale == 0 || foundItem?.scale == 1) { // library marker kkossev.deviceProfileLib, line 1031 convertedValue = tuyaValue as double // library marker kkossev.deviceProfileLib, line 1032 } // library marker kkossev.deviceProfileLib, line 1033 else { // library marker kkossev.deviceProfileLib, line 1034 convertedValue = (tuyaValue as double) / (foundItem.scale as double) // library marker kkossev.deviceProfileLib, line 1035 } // library marker kkossev.deviceProfileLib, line 1036 isEqual = Math.abs((convertedValue as double) - (hubitatValue as double)) < 0.001 // library marker kkossev.deviceProfileLib, line 1037 logTrace "compareAndConvertDecimals: tuyaValue=${tuyaValue} foundItem.scale=${foundItem.scale} convertedValue=${convertedValue} to hubitatValue=${hubitatValue} isEqual=${isEqual}" // library marker kkossev.deviceProfileLib, line 1038 return [isEqual, convertedValue] // library marker kkossev.deviceProfileLib, line 1039 } // library marker kkossev.deviceProfileLib, line 1040 /* groovylint-disable-next-line MethodParameterTypeRequired, NoDef */ // library marker kkossev.deviceProfileLib, line 1042 private List compareAndConvertEnumKeys(final Map foundItem, int tuyaValue, hubitatValue) { // library marker kkossev.deviceProfileLib, line 1043 //logTrace "compareAndConvertEnumKeys: tuyaValue=${tuyaValue} hubitatValue=${hubitatValue}" // library marker kkossev.deviceProfileLib, line 1044 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 1045 def convertedValue // library marker kkossev.deviceProfileLib, line 1046 if (foundItem?.scale == null || foundItem?.scale == 0 || foundItem?.scale == 1) { // library marker kkossev.deviceProfileLib, line 1047 convertedValue = tuyaValue as int // library marker kkossev.deviceProfileLib, line 1048 isEqual = ((convertedValue as int) == (safeToInt(hubitatValue))) // library marker kkossev.deviceProfileLib, line 1049 } // library marker kkossev.deviceProfileLib, line 1050 else { // scaled value - divide by scale // library marker kkossev.deviceProfileLib, line 1051 double hubitatSafeValue = safeToDouble(hubitatValue, -1.0) // library marker kkossev.deviceProfileLib, line 1052 convertedValue = (tuyaValue as double) / (foundItem.scale as double) // library marker kkossev.deviceProfileLib, line 1053 if (hubitatSafeValue == -1.0) { // library marker kkossev.deviceProfileLib, line 1054 isEqual = false // library marker kkossev.deviceProfileLib, line 1055 } // library marker kkossev.deviceProfileLib, line 1056 else { // compare as double (float) // library marker kkossev.deviceProfileLib, line 1057 isEqual = Math.abs((convertedValue as double) - (hubitatSafeValue as double)) < 0.001 // library marker kkossev.deviceProfileLib, line 1058 } // library marker kkossev.deviceProfileLib, line 1059 } // library marker kkossev.deviceProfileLib, line 1060 //logTrace "compareAndConvertEnumKeys: tuyaValue=${tuyaValue} foundItem.scale=${foundItem.scale} convertedValue=${convertedValue} to hubitatValue=${hubitatValue} isEqual=${isEqual}" // library marker kkossev.deviceProfileLib, line 1061 return [isEqual, convertedValue] // library marker kkossev.deviceProfileLib, line 1062 } // library marker kkossev.deviceProfileLib, line 1063 /* groovylint-disable-next-line MethodParameterTypeRequired, NoDef */ // library marker kkossev.deviceProfileLib, line 1065 private List compareAndConvertTuyaToHubitatPreferenceValue(final Map foundItem, fncmd, preference) { // library marker kkossev.deviceProfileLib, line 1066 if (foundItem == null || fncmd == null || preference == null) { return [true, 'none'] } // library marker kkossev.deviceProfileLib, line 1067 if (foundItem?.type == null) { return [true, 'none'] } // library marker kkossev.deviceProfileLib, line 1068 boolean isEqual // library marker kkossev.deviceProfileLib, line 1069 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 1070 def tuyaValueScaled // could be integer or float // library marker kkossev.deviceProfileLib, line 1071 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 1072 def preferenceValue = settings[foundItem.name] // library marker kkossev.deviceProfileLib, line 1073 switch (foundItem.type) { // library marker kkossev.deviceProfileLib, line 1074 case 'bool' : // [0:"OFF", 1:"ON"] // library marker kkossev.deviceProfileLib, line 1075 (isEqual, tuyaValueScaled) = compareAndConvertNumbers(foundItem, safeToInt(fncmd), safeToInt(preference)) // library marker kkossev.deviceProfileLib, line 1076 logTrace "compareAndConvertTuyaToHubitatPreferenceValue: bool: preference = ${preference} type=${foundItem.type} foundItem=${foundItem.name} isEqual=${isEqual} preferenceValue=${preferenceValue} tuyaValueScaled=${tuyaValueScaled} fncmd=${fncmd}" // library marker kkossev.deviceProfileLib, line 1077 break // library marker kkossev.deviceProfileLib, line 1078 case 'enum' : // [0:"inactive", 1:"active"] map:['75': '0.75 meters', '150': '1.50 meters', '225': '2.25 meters'] // library marker kkossev.deviceProfileLib, line 1079 Integer scale = (foundItem.scale ?: 0 ) as int // library marker kkossev.deviceProfileLib, line 1080 if (scale != null && scale != 0 && scale != 1) { // library marker kkossev.deviceProfileLib, line 1081 preferenceValue = preferenceValue.toString().replace('[', '').replace(']', '') // library marker kkossev.deviceProfileLib, line 1082 /* groovylint-disable-next-line ParameterReassignment */ // library marker kkossev.deviceProfileLib, line 1083 preference = preference.toString().replace('[', '').replace(']', '') // library marker kkossev.deviceProfileLib, line 1084 logTrace "compareAndConvertTuyaToHubitatPreferenceValue: enum: scale=${scale} fncmd=${fncmd} preference=${preference} preferenceValue=${preferenceValue} safeToDouble(fncmd)=${safeToDouble(fncmd)} safeToDouble(preference)=${safeToDouble(preference)}" // library marker kkossev.deviceProfileLib, line 1085 (isEqual, tuyaValueScaled) = compareAndConvertDecimals(foundItem, safeToDouble(fncmd), safeToDouble(preference)) // library marker kkossev.deviceProfileLib, line 1086 } // library marker kkossev.deviceProfileLib, line 1087 else { // library marker kkossev.deviceProfileLib, line 1088 (isEqual, tuyaValueScaled) = compareAndConvertNumbers(foundItem, safeToInt(fncmd), safeToInt(preference)) // library marker kkossev.deviceProfileLib, line 1089 } // library marker kkossev.deviceProfileLib, line 1090 logTrace "compareAndConvertTuyaToHubitatPreferenceValue: enum: preference = ${preference} type=${foundItem.type} foundItem=${foundItem.name} isEqual=${isEqual} preferenceValue=${preferenceValue} tuyaValueScaled=${tuyaValueScaled} fncmd=${fncmd}" // library marker kkossev.deviceProfileLib, line 1091 break // library marker kkossev.deviceProfileLib, line 1092 case 'value' : // depends on foundItem.scale // library marker kkossev.deviceProfileLib, line 1093 case 'number' : // library marker kkossev.deviceProfileLib, line 1094 (isEqual, tuyaValueScaled) = compareAndConvertNumbers(foundItem, safeToInt(fncmd), safeToInt(preference)) // library marker kkossev.deviceProfileLib, line 1095 logTrace "tuyaValue=${tuyaValue} tuyaValueScaled=${tuyaValueScaled} preferenceValue = ${preference} isEqual=${isEqual}" // library marker kkossev.deviceProfileLib, line 1096 break // library marker kkossev.deviceProfileLib, line 1097 case 'decimal' : // library marker kkossev.deviceProfileLib, line 1098 (isEqual, tuyaValueScaled) = compareAndConvertDecimals(foundItem, safeToDouble(fncmd), safeToDouble(preference)) // library marker kkossev.deviceProfileLib, line 1099 logTrace "comparing as float tuyaValue=${tuyaValue} foundItem.scale=${foundItem.scale} tuyaValueScaled=${tuyaValueScaled} to preferenceValue = ${preference}" // library marker kkossev.deviceProfileLib, line 1100 break // library marker kkossev.deviceProfileLib, line 1101 default : // library marker kkossev.deviceProfileLib, line 1102 logDebug 'compareAndConvertTuyaToHubitatPreferenceValue: unsupported type %{foundItem.type}' // library marker kkossev.deviceProfileLib, line 1103 return [true, 'none'] // fallback - assume equal // library marker kkossev.deviceProfileLib, line 1104 } // library marker kkossev.deviceProfileLib, line 1105 if (isEqual == false) { // library marker kkossev.deviceProfileLib, line 1106 logDebug "compareAndConvertTuyaToHubitatPreferenceValue: preference = ${preference} type=${foundItem.type} foundItem=${foundItem.name} isEqual=${isEqual} tuyaValueScaled=${tuyaValueScaled} (scale=${foundItem.scale}) fncmd=${fncmd}" // library marker kkossev.deviceProfileLib, line 1107 } // library marker kkossev.deviceProfileLib, line 1108 // // library marker kkossev.deviceProfileLib, line 1109 return [isEqual, tuyaValueScaled] // library marker kkossev.deviceProfileLib, line 1110 } // library marker kkossev.deviceProfileLib, line 1111 // // library marker kkossev.deviceProfileLib, line 1113 // called from process TuyaDP from DeviceProfile() // library marker kkossev.deviceProfileLib, line 1114 // compares the value of the DP foundItem against a Preference with the same name // library marker kkossev.deviceProfileLib, line 1115 // returns: (two results!) // library marker kkossev.deviceProfileLib, line 1116 // isEqual : true - if the Tuya DP value equals to the DP calculated value (no need to update the preference) // library marker kkossev.deviceProfileLib, line 1117 // : true - if a preference with the same name does not exist (no preference value to update) // library marker kkossev.deviceProfileLib, line 1118 // isEqual : false - the reported DP value is different than the corresponding preference (the preference needs to be updated!) // library marker kkossev.deviceProfileLib, line 1119 // // library marker kkossev.deviceProfileLib, line 1120 // hubitatEventValue - the converted DP value, scaled (divided by the scale factor) to match the corresponding preference type value // library marker kkossev.deviceProfileLib, line 1121 // // library marker kkossev.deviceProfileLib, line 1122 // TODO: refactor! // library marker kkossev.deviceProfileLib, line 1123 // // library marker kkossev.deviceProfileLib, line 1124 /* groovylint-disable-next-line MethodParameterTypeRequired, NoDef, UnusedMethodParameter */ // library marker kkossev.deviceProfileLib, line 1125 private List compareAndConvertTuyaToHubitatEventValue(Map foundItem, int fncmd, boolean doNotTrace=false) { // library marker kkossev.deviceProfileLib, line 1126 if (foundItem == null) { return [true, 'none'] } // library marker kkossev.deviceProfileLib, line 1127 if (foundItem.type == null) { return [true, 'none'] } // library marker kkossev.deviceProfileLib, line 1128 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 1129 def hubitatEventValue // could be integer or float or string // library marker kkossev.deviceProfileLib, line 1130 boolean isEqual // library marker kkossev.deviceProfileLib, line 1131 switch (foundItem.type) { // library marker kkossev.deviceProfileLib, line 1132 case 'bool' : // [0:"OFF", 1:"ON"] // library marker kkossev.deviceProfileLib, line 1133 (isEqual, hubitatEventValue) = compareAndConvertStrings(foundItem, foundItem.map[fncmd as int] ?: 'unknown', device.currentValue(foundItem.name) ?: 'unknown') // library marker kkossev.deviceProfileLib, line 1134 break // library marker kkossev.deviceProfileLib, line 1135 case 'enum' : // [0:"inactive", 1:"active"] foundItem.map=[75:0.75 meters, 150:1.50 meters, 225:2.25 meters, 300:3.00 meters, 375:3.75 meters, 450:4.50 meters] // library marker kkossev.deviceProfileLib, line 1136 logTrace "compareAndConvertTuyaToHubitatEventValue: enum: foundItem.scale=${foundItem.scale}, fncmd=${fncmd}, device.currentValue(${foundItem.name})=${(device.currentValue(foundItem.name))} map=${foundItem.map}" // library marker kkossev.deviceProfileLib, line 1137 Object latestEvent = device.currentState(foundItem.name) // library marker kkossev.deviceProfileLib, line 1138 String dataType = latestEvent?.dataType // library marker kkossev.deviceProfileLib, line 1139 logTrace "latestEvent is ${latestEvent} dataType is ${dataType}" // library marker kkossev.deviceProfileLib, line 1140 // if the attribute is of a type enum, the value is a string. Compare the string values! // library marker kkossev.deviceProfileLib, line 1141 if (dataType == null || dataType == 'ENUM') { // library marker kkossev.deviceProfileLib, line 1142 (isEqual, hubitatEventValue) = compareAndConvertStrings(foundItem, foundItem.map[fncmd as int] ?: 'unknown', device.currentValue(foundItem.name) ?: 'unknown') // library marker kkossev.deviceProfileLib, line 1143 } // library marker kkossev.deviceProfileLib, line 1144 else { // library marker kkossev.deviceProfileLib, line 1145 (isEqual, hubitatEventValue) = compareAndConvertEnumKeys(foundItem, fncmd, device.currentValue(foundItem.name)) // library marker kkossev.deviceProfileLib, line 1146 } // library marker kkossev.deviceProfileLib, line 1147 logTrace "compareAndConvertTuyaToHubitatEventValue: after compareAndConvertStrings: isEqual=${isEqual} hubitatEventValue=${hubitatEventValue}" // library marker kkossev.deviceProfileLib, line 1148 break // library marker kkossev.deviceProfileLib, line 1149 case 'value' : // depends on foundItem.scale // library marker kkossev.deviceProfileLib, line 1150 case 'number' : // library marker kkossev.deviceProfileLib, line 1151 //logTrace "compareAndConvertTuyaToHubitatEventValue: foundItem.scale=${foundItem.scale} fncmd=${fncmd} device.currentValue(${foundItem.name})=${(device.currentValue(foundItem.name))}" // library marker kkossev.deviceProfileLib, line 1152 (isEqual, hubitatEventValue) = compareAndConvertNumbers(foundItem, safeToInt(fncmd), safeToInt(device.currentValue(foundItem.name))) // library marker kkossev.deviceProfileLib, line 1153 break // library marker kkossev.deviceProfileLib, line 1154 case 'decimal' : // library marker kkossev.deviceProfileLib, line 1155 (isEqual, hubitatEventValue) = compareAndConvertDecimals(foundItem, safeToDouble(fncmd), safeToDouble(device.currentValue(foundItem.name))) // library marker kkossev.deviceProfileLib, line 1156 break // library marker kkossev.deviceProfileLib, line 1157 default : // library marker kkossev.deviceProfileLib, line 1158 logDebug 'compareAndConvertTuyaToHubitatEventValue: unsupported dpType %{foundItem.type}' // library marker kkossev.deviceProfileLib, line 1159 return [true, 'none'] // fallback - assume equal // library marker kkossev.deviceProfileLib, line 1160 } // library marker kkossev.deviceProfileLib, line 1161 //if (!doNotTrace) log.trace "foundItem=${foundItem.name} isEqual=${isEqual} attrValue=${attrValue} fncmd=${fncmd} foundItem.scale=${foundItem.scale } valueScaled=${valueScaled} " // library marker kkossev.deviceProfileLib, line 1162 return [isEqual, hubitatEventValue] // library marker kkossev.deviceProfileLib, line 1163 } // library marker kkossev.deviceProfileLib, line 1164 public Integer preProc(final Map foundItem, int fncmd_orig) { // library marker kkossev.deviceProfileLib, line 1166 Integer fncmd = fncmd_orig // library marker kkossev.deviceProfileLib, line 1167 if (foundItem == null) { return fncmd } // library marker kkossev.deviceProfileLib, line 1168 if (foundItem.preProc == null) { return fncmd } // library marker kkossev.deviceProfileLib, line 1169 String preProcFunction = foundItem.preProc // library marker kkossev.deviceProfileLib, line 1170 //logDebug "preProc: foundItem.preProc = ${preProcFunction}" // library marker kkossev.deviceProfileLib, line 1171 // check if preProc method exists // library marker kkossev.deviceProfileLib, line 1172 if (!this.respondsTo(preProcFunction)) { // library marker kkossev.deviceProfileLib, line 1173 logDebug "preProc: function ${preProcFunction} not found" // library marker kkossev.deviceProfileLib, line 1174 return fncmd_orig // library marker kkossev.deviceProfileLib, line 1175 } // library marker kkossev.deviceProfileLib, line 1176 // execute the preProc function // library marker kkossev.deviceProfileLib, line 1177 try { // library marker kkossev.deviceProfileLib, line 1178 fncmd = "$preProcFunction"(fncmd_orig) // library marker kkossev.deviceProfileLib, line 1179 } // library marker kkossev.deviceProfileLib, line 1180 catch (e) { // library marker kkossev.deviceProfileLib, line 1181 logWarn "preProc: Exception '${e}' caught while processing $preProcFunction($fncmd_orig) (val=${fncmd}))" // library marker kkossev.deviceProfileLib, line 1182 return fncmd_orig // library marker kkossev.deviceProfileLib, line 1183 } // library marker kkossev.deviceProfileLib, line 1184 //logDebug "setFunction result is ${fncmd}" // library marker kkossev.deviceProfileLib, line 1185 return fncmd // library marker kkossev.deviceProfileLib, line 1186 } // library marker kkossev.deviceProfileLib, line 1187 // TODO: refactor! // library marker kkossev.deviceProfileLib, line 1189 // called from custom drivers (customParseE002Cluster customParseFC11Cluster customParseOccupancyCluster ...) // library marker kkossev.deviceProfileLib, line 1190 // returns true if the DP was processed successfully, false otherwise. // library marker kkossev.deviceProfileLib, line 1191 public boolean processClusterAttributeFromDeviceProfile(final Map descMap) { // library marker kkossev.deviceProfileLib, line 1192 logTrace "processClusterAttributeFromDeviceProfile: descMap = ${descMap}" // library marker kkossev.deviceProfileLib, line 1193 if (state.deviceProfile == null) { logTrace 'state.deviceProfile is missing!'; return false } // library marker kkossev.deviceProfileLib, line 1194 if (descMap == null || descMap == [:] || descMap.cluster == null || descMap.attrId == null || descMap.value == null) { logTrace 'descMap is missing cluster, attribute or value!'; return false } // library marker kkossev.deviceProfileLib, line 1195 List attribMap = deviceProfilesV3[state.deviceProfile]?.attributes // library marker kkossev.deviceProfileLib, line 1197 if (attribMap == null || attribMap?.isEmpty()) { return false } // no any attributes are defined in the Device Profile // library marker kkossev.deviceProfileLib, line 1198 String clusterAttribute = "0x${descMap.cluster}:0x${descMap.attrId}" // library marker kkossev.deviceProfileLib, line 1200 int value // library marker kkossev.deviceProfileLib, line 1201 try { // library marker kkossev.deviceProfileLib, line 1202 value = hexStrToUnsignedInt(descMap.value) // library marker kkossev.deviceProfileLib, line 1203 } // library marker kkossev.deviceProfileLib, line 1204 catch (e) { // library marker kkossev.deviceProfileLib, line 1205 logWarn "processClusterAttributeFromDeviceProfile: exception ${e} caught while converting hex value ${descMap.value} to integer" // library marker kkossev.deviceProfileLib, line 1206 return false // library marker kkossev.deviceProfileLib, line 1207 } // library marker kkossev.deviceProfileLib, line 1208 Map foundItem = attribMap.find { it['at'] == clusterAttribute } // library marker kkossev.deviceProfileLib, line 1209 if (foundItem == null || foundItem == [:]) { // library marker kkossev.deviceProfileLib, line 1210 // clusterAttribute was not found into the attributes list for this particular deviceProfile // library marker kkossev.deviceProfileLib, line 1211 // updateStateUnknownclusterAttribute(descMap) // library marker kkossev.deviceProfileLib, line 1212 // continue processing the descMap report in the old code ... // library marker kkossev.deviceProfileLib, line 1213 logTrace "processClusterAttributeFromDeviceProfile: clusterAttribute ${clusterAttribute} was not found in the attributes list for this deviceProfile ${DEVICE?.description}" // library marker kkossev.deviceProfileLib, line 1214 return false // library marker kkossev.deviceProfileLib, line 1215 } // library marker kkossev.deviceProfileLib, line 1216 value = convertSignedInts(value, foundItem) // library marker kkossev.deviceProfileLib, line 1217 return processFoundItem(descMap, foundItem, value, isSpammyDPsToNotTrace(descMap)) // library marker kkossev.deviceProfileLib, line 1218 } // library marker kkossev.deviceProfileLib, line 1219 /** // library marker kkossev.deviceProfileLib, line 1221 * Called from standardProcessTuyaDP method in commonLib // library marker kkossev.deviceProfileLib, line 1222 * // library marker kkossev.deviceProfileLib, line 1223 * Processes a Tuya DP (Data Point) received from the device, based on the device profile and its defined Tuya DPs. // library marker kkossev.deviceProfileLib, line 1224 * If a preference exists for the DP, it updates the preference value and sends an event if the DP is declared as an attribute. // library marker kkossev.deviceProfileLib, line 1225 * If no preference exists for the DP, it logs the DP value as an info message. // library marker kkossev.deviceProfileLib, line 1226 * If the DP is spammy (not needed for anything), it does not perform any further processing. // library marker kkossev.deviceProfileLib, line 1227 * // library marker kkossev.deviceProfileLib, line 1228 * @return true if the DP was processed successfully, false otherwise. // library marker kkossev.deviceProfileLib, line 1229 */ // library marker kkossev.deviceProfileLib, line 1230 /* groovylint-disable-next-line UnusedMethodParameter */ // library marker kkossev.deviceProfileLib, line 1231 public boolean processTuyaDPfromDeviceProfile(final Map descMap, final int dp, final int dp_id, final int fncmd_orig, final int dp_len) { // library marker kkossev.deviceProfileLib, line 1232 int fncmd = fncmd_orig // library marker kkossev.deviceProfileLib, line 1233 if (state.deviceProfile == null) { return false } // library marker kkossev.deviceProfileLib, line 1234 if (isSpammyDPsToIgnore(descMap)) { return true } // do not perform any further processing, if this is a spammy report that is not needed for anyhting (such as the LED status) // library marker kkossev.deviceProfileLib, line 1235 List tuyaDPsMap = deviceProfilesV3[state.deviceProfile]?.tuyaDPs // library marker kkossev.deviceProfileLib, line 1237 if (tuyaDPsMap == null || tuyaDPsMap == [:]) { return false } // no any Tuya DPs defined in the Device Profile // library marker kkossev.deviceProfileLib, line 1238 Map foundItem = tuyaDPsMap.find { it['dp'] == (dp as int) } // library marker kkossev.deviceProfileLib, line 1240 if (foundItem == null || foundItem == [:]) { // library marker kkossev.deviceProfileLib, line 1241 // DP was not found into the tuyaDPs list for this particular deviceProfile // library marker kkossev.deviceProfileLib, line 1242 // updateStateUnknownDPs(descMap, dp, dp_id, fncmd, dp_len) // TODO !!!!!!!!!!!!!!!!!!!!!!!!!!!! // library marker kkossev.deviceProfileLib, line 1243 // continue processing the DP report in the old code ... // library marker kkossev.deviceProfileLib, line 1244 return false // library marker kkossev.deviceProfileLib, line 1245 } // library marker kkossev.deviceProfileLib, line 1246 return processFoundItem(descMap, foundItem, fncmd, isSpammyDPsToNotTrace(descMap)) // library marker kkossev.deviceProfileLib, line 1247 } // library marker kkossev.deviceProfileLib, line 1248 /* // library marker kkossev.deviceProfileLib, line 1250 * deviceProfile DP processor : updates the preference value and calls a custom handler or sends an event if the DP is declared as an attribute in the device profile // library marker kkossev.deviceProfileLib, line 1251 */ // library marker kkossev.deviceProfileLib, line 1252 private boolean processFoundItem(final Map descMap, final Map foundItem, int value, boolean doNotTrace = false) { // library marker kkossev.deviceProfileLib, line 1253 if (foundItem == null) { return false } // library marker kkossev.deviceProfileLib, line 1254 // added 10/31/2023 - preProc the attribute value if needed // library marker kkossev.deviceProfileLib, line 1255 if (foundItem.preProc != null) { // library marker kkossev.deviceProfileLib, line 1256 /* groovylint-disable-next-line ParameterReassignment */ // library marker kkossev.deviceProfileLib, line 1257 Integer preProcValue = preProc(foundItem, value) // library marker kkossev.deviceProfileLib, line 1258 if (preProcValue == null) { logDebug "processFoundItem: preProc returned null for ${foundItem.name} value ${value} -> further processing is skipped!" ; return true } // library marker kkossev.deviceProfileLib, line 1259 if (preProcValue != value) { // library marker kkossev.deviceProfileLib, line 1260 logDebug "processFoundItem: preProc changed ${foundItem.name} value to ${preProcValue}" // library marker kkossev.deviceProfileLib, line 1261 /* groovylint-disable-next-line ParameterReassignment */ // library marker kkossev.deviceProfileLib, line 1262 value = preProcValue as int // library marker kkossev.deviceProfileLib, line 1263 } // library marker kkossev.deviceProfileLib, line 1264 } // library marker kkossev.deviceProfileLib, line 1265 else { logTrace "processFoundItem: no preProc for ${foundItem.name}" } // library marker kkossev.deviceProfileLib, line 1266 String name = foundItem.name // preference name as in the attributes map // library marker kkossev.deviceProfileLib, line 1268 String existingPrefValue = settings[foundItem.name] ?: 'none' // existing preference value // library marker kkossev.deviceProfileLib, line 1269 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 1270 def preferenceValue = null // preference value // library marker kkossev.deviceProfileLib, line 1271 //log.trace "settings=${settings}" // library marker kkossev.deviceProfileLib, line 1272 boolean preferenceExists = (DEVICE?.preferences != null && !DEVICE?.preferences?.isEmpty()) ? DEVICE?.preferences?.containsKey(foundItem.name) : false // check if there is an existing preference for this clusterAttribute // library marker kkossev.deviceProfileLib, line 1273 //log.trace "preferenceExists=${preferenceExists}" // library marker kkossev.deviceProfileLib, line 1274 boolean isAttribute = device.hasAttribute(foundItem.name) // check if there is such a attribute for this clusterAttribute // library marker kkossev.deviceProfileLib, line 1275 boolean isEqual = false // library marker kkossev.deviceProfileLib, line 1276 boolean wasChanged = false // library marker kkossev.deviceProfileLib, line 1277 if (!doNotTrace) { logTrace "processFoundItem: name=${foundItem.name}, isAttribute=${isAttribute}, preferenceExists=${preferenceExists}, existingPrefValue=${existingPrefValue} (type ${foundItem.type}, rw=${foundItem.rw}) value is ${value} (description: ${foundItem.description})" } // library marker kkossev.deviceProfileLib, line 1278 // check if the clusterAttribute has the same value as the last one, or the value has changed // library marker kkossev.deviceProfileLib, line 1279 // the previous value may be stored in an attribute, as a preference, as both attribute and preference or not stored anywhere ... // library marker kkossev.deviceProfileLib, line 1280 String unitText = foundItem.unit != null ? "$foundItem.unit" : '' // library marker kkossev.deviceProfileLib, line 1281 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 1282 def valueScaled // can be number or decimal or string // library marker kkossev.deviceProfileLib, line 1283 String descText = descText = "${name} is ${value} ${unitText}" // the default description text for log events // library marker kkossev.deviceProfileLib, line 1284 // TODO - check if clusterAttribute is in the list of the received state.attributes - then we have something to compare ! // library marker kkossev.deviceProfileLib, line 1286 if (!isAttribute && !preferenceExists) { // if the previous value of this clusterAttribute is not stored anywhere - just seend an Info log if Debug is enabled // library marker kkossev.deviceProfileLib, line 1287 if (!doNotTrace) { // only if the clusterAttribute is not in the spammy list // library marker kkossev.deviceProfileLib, line 1288 logTrace "processFoundItem: no preference or attribute for ${name} - just log the value, if not equal to the last one..." // library marker kkossev.deviceProfileLib, line 1289 // TODO - scaledValue ????? TODO! // library marker kkossev.deviceProfileLib, line 1290 descText = "${name} is ${value} ${unitText}" // library marker kkossev.deviceProfileLib, line 1291 if (settings.logEnable) { logInfo "${descText} (Debug logging is enabled)" } // only when Debug is enabled! // library marker kkossev.deviceProfileLib, line 1292 } // library marker kkossev.deviceProfileLib, line 1293 return true // no more processing is needed, as this clusterAttribute is NOT a preference and NOT an attribute // library marker kkossev.deviceProfileLib, line 1294 } // library marker kkossev.deviceProfileLib, line 1295 // first, check if there is a preference defined in the deviceProfileV3 to be updated // library marker kkossev.deviceProfileLib, line 1297 if (preferenceExists && !doNotTrace) { // do not even try to automatically update the preference if it is in the spammy list! - added 04/23/2024 // library marker kkossev.deviceProfileLib, line 1298 // preference exists and its's value is extracted // library marker kkossev.deviceProfileLib, line 1299 (isEqual, preferenceValue) = compareAndConvertTuyaToHubitatPreferenceValue(foundItem, value, existingPrefValue) // library marker kkossev.deviceProfileLib, line 1300 logTrace "processFoundItem: preference '${name}' exists with existingPrefValue ${existingPrefValue} (type ${foundItem.type}) -> isEqual=${isEqual} preferenceValue=${preferenceValue}" // library marker kkossev.deviceProfileLib, line 1301 if (isEqual == true) { // the preference is not changed - do nothing // library marker kkossev.deviceProfileLib, line 1302 //log.trace "doNotTrace=${doNotTrace} isSpammyDeviceProfile=${isSpammyDeviceProfile()}" // library marker kkossev.deviceProfileLib, line 1303 if (!(doNotTrace || isSpammyDeviceProfile())) { // the clusterAttribute value is the same as the preference value - no need to update the preference // library marker kkossev.deviceProfileLib, line 1304 logDebug "processFoundItem: no change: preference '${name}' existingPrefValue ${existingPrefValue} equals scaled value ${preferenceValue} (clusterAttribute raw value ${value})" // library marker kkossev.deviceProfileLib, line 1305 } // library marker kkossev.deviceProfileLib, line 1306 } // library marker kkossev.deviceProfileLib, line 1307 else { // the preferences has changed - update it! // library marker kkossev.deviceProfileLib, line 1308 String scaledPreferenceValue = preferenceValue // library marker kkossev.deviceProfileLib, line 1309 if (foundItem.type == 'enum' && foundItem.scale != null && foundItem.scale != 0 && foundItem.scale != 1) { // library marker kkossev.deviceProfileLib, line 1310 scaledPreferenceValue = ((preferenceValue * safeToInt(foundItem.scale)) as int).toString() // library marker kkossev.deviceProfileLib, line 1311 } // library marker kkossev.deviceProfileLib, line 1312 logDebug "processFoundItem: preference '${name}' value ${existingPrefValue} differs from the new scaled value ${preferenceValue} (clusterAttribute raw value ${value})" // library marker kkossev.deviceProfileLib, line 1313 if (settings.logEnable) { logInfo "updating the preference '${name}' from ${existingPrefValue} to ${preferenceValue} (scaledPreferenceValue=${scaledPreferenceValue}, type=${foundItem.type})" } // library marker kkossev.deviceProfileLib, line 1314 try { // library marker kkossev.deviceProfileLib, line 1315 device.updateSetting("${name}", [value:scaledPreferenceValue, type:foundItem.type]) // library marker kkossev.deviceProfileLib, line 1316 wasChanged = true // library marker kkossev.deviceProfileLib, line 1317 } // library marker kkossev.deviceProfileLib, line 1318 catch (e) { // library marker kkossev.deviceProfileLib, line 1319 logWarn "exception ${e} caught while updating preference ${name} to ${preferenceValue}, type ${foundItem.type}" // library marker kkossev.deviceProfileLib, line 1320 } // library marker kkossev.deviceProfileLib, line 1321 } // library marker kkossev.deviceProfileLib, line 1322 } // library marker kkossev.deviceProfileLib, line 1323 else { // no preference exists for this clusterAttribute // library marker kkossev.deviceProfileLib, line 1324 // if not in the spammy list - log it! // library marker kkossev.deviceProfileLib, line 1325 unitText = foundItem.unit != null ? "$foundItem.unit" : '' // TODO - check if unitText must be declared here or outside the if block // library marker kkossev.deviceProfileLib, line 1326 //logInfo "${name} is ${value} ${unitText}" // library marker kkossev.deviceProfileLib, line 1327 } // library marker kkossev.deviceProfileLib, line 1328 // second, send an event if this is declared as an attribute! // library marker kkossev.deviceProfileLib, line 1330 if (isAttribute) { // this clusterAttribute has an attribute that must be sent in an Event // library marker kkossev.deviceProfileLib, line 1331 (isEqual, valueScaled) = compareAndConvertTuyaToHubitatEventValue(foundItem, value, doNotTrace) // library marker kkossev.deviceProfileLib, line 1332 if (isEqual == false) { logTrace "attribute '${name}' exists (type ${foundItem.type}), value ${value} -> isEqual=${isEqual} valueScaled=${valueScaled} wasChanged=${wasChanged}" } // library marker kkossev.deviceProfileLib, line 1333 descText = "${name} is ${valueScaled} ${unitText}" // library marker kkossev.deviceProfileLib, line 1334 if (settings?.logEnable == true) { descText += " (raw:${value})" } // library marker kkossev.deviceProfileLib, line 1335 if (state.states != null && state.states['isRefresh'] == true) { descText += ' [refresh]' } // library marker kkossev.deviceProfileLib, line 1336 if (isEqual && !wasChanged) { // this DP report has the same value as the last one - just send a debug log and move along! // library marker kkossev.deviceProfileLib, line 1337 if (!doNotTrace) { // library marker kkossev.deviceProfileLib, line 1338 if (settings.logEnable) { logDebug "${descText } (no change)" } // library marker kkossev.deviceProfileLib, line 1339 } // library marker kkossev.deviceProfileLib, line 1340 if (foundItem.processDuplicated == true) { // library marker kkossev.deviceProfileLib, line 1341 logDebug 'processDuplicated=true -> continue' // library marker kkossev.deviceProfileLib, line 1342 } // library marker kkossev.deviceProfileLib, line 1343 // patch for inverted motion sensor 2-in-1 // library marker kkossev.deviceProfileLib, line 1345 if (name == 'motion' && is2in1()) { // TODO - remove the patch !! // library marker kkossev.deviceProfileLib, line 1346 logDebug 'patch for inverted motion sensor 2-in-1' // library marker kkossev.deviceProfileLib, line 1347 // continue ... // library marker kkossev.deviceProfileLib, line 1348 } // library marker kkossev.deviceProfileLib, line 1349 else { // library marker kkossev.deviceProfileLib, line 1351 if (state.states != null && state.states['isRefresh'] == true) { // library marker kkossev.deviceProfileLib, line 1352 logTrace 'isRefresh = true - continue and send an event, although there was no change...' // library marker kkossev.deviceProfileLib, line 1353 } // library marker kkossev.deviceProfileLib, line 1354 else { // library marker kkossev.deviceProfileLib, line 1355 //log.trace "should not be here !!!!!!!!!!" // library marker kkossev.deviceProfileLib, line 1356 return true // we are done (if there was potentially a preference, it should be already set to the same value) // library marker kkossev.deviceProfileLib, line 1357 } // library marker kkossev.deviceProfileLib, line 1358 } // library marker kkossev.deviceProfileLib, line 1359 } // library marker kkossev.deviceProfileLib, line 1360 // clusterAttribute value (value) is not equal to the attribute last value or was changed- we must send an update event! // library marker kkossev.deviceProfileLib, line 1362 int divider = safeToInt(foundItem.scale ?: 1) ?: 1 // library marker kkossev.deviceProfileLib, line 1363 float valueCorrected = value / divider // library marker kkossev.deviceProfileLib, line 1364 if (!doNotTrace) { logTrace "value=${value} foundItem.scale=${foundItem.scale} divider=${divider} valueCorrected=${valueCorrected}" } // library marker kkossev.deviceProfileLib, line 1365 // process the events in the device specific driver.. // library marker kkossev.deviceProfileLib, line 1366 if (this.respondsTo('customProcessDeviceProfileEvent')) { // library marker kkossev.deviceProfileLib, line 1367 customProcessDeviceProfileEvent(descMap, name, valueScaled, unitText, descText) // used in Zigbee_TRV // library marker kkossev.deviceProfileLib, line 1368 } // library marker kkossev.deviceProfileLib, line 1369 else { // library marker kkossev.deviceProfileLib, line 1370 // no custom handler - send the event as usual // library marker kkossev.deviceProfileLib, line 1371 sendEvent(name : name, value : valueScaled, unit:unitText, descriptionText: descText, type: 'physical', isStateChange: true) // attribute value is changed - send an event ! // library marker kkossev.deviceProfileLib, line 1372 if (!doNotTrace) { // library marker kkossev.deviceProfileLib, line 1373 logTrace "event ${name} sent w/ valueScaled ${valueScaled}" // library marker kkossev.deviceProfileLib, line 1374 logInfo "${descText}" // TODO - send info log only if the value has changed? // TODO - check whether Info log will be sent also for spammy clusterAttribute ? // library marker kkossev.deviceProfileLib, line 1375 } // library marker kkossev.deviceProfileLib, line 1376 } // library marker kkossev.deviceProfileLib, line 1377 } // library marker kkossev.deviceProfileLib, line 1378 return true // all processing was done here! // library marker kkossev.deviceProfileLib, line 1379 } // library marker kkossev.deviceProfileLib, line 1380 // not used ? (except for debugging)? TODO // library marker kkossev.deviceProfileLib, line 1382 public boolean validateAndFixPreferences(String debugStr) { return validateAndFixPreferences(debugStr.toBoolean() as boolean) } // library marker kkossev.deviceProfileLib, line 1383 public boolean validateAndFixPreferences(boolean debug=false) { // library marker kkossev.deviceProfileLib, line 1384 //debug = true // library marker kkossev.deviceProfileLib, line 1385 if (debug) { logTrace "validateAndFixPreferences: preferences=${DEVICE?.preferences}" } // library marker kkossev.deviceProfileLib, line 1386 if (DEVICE?.preferences == null || DEVICE?.preferences == [:]) { logDebug "validateAndFixPreferences: no preferences defined for device profile ${getDeviceProfile()}" ; return false } // library marker kkossev.deviceProfileLib, line 1387 int validationFailures = 0, validationFixes = 0, total = 0 // library marker kkossev.deviceProfileLib, line 1388 /* groovylint-disable-next-line NoDef, VariableTypeRequired */ // library marker kkossev.deviceProfileLib, line 1389 def oldSettingValue, newValue // library marker kkossev.deviceProfileLib, line 1390 String settingType = '' // library marker kkossev.deviceProfileLib, line 1391 DEVICE?.preferences.each { // library marker kkossev.deviceProfileLib, line 1392 Map foundMap = getPreferencesMapByName(it.key) // library marker kkossev.deviceProfileLib, line 1393 if (foundMap == null || foundMap == [:]) { logDebug "validateAndFixPreferences: map not found for preference ${it.key}" ; return false } // library marker kkossev.deviceProfileLib, line 1394 settingType = device.getSettingType(it.key) ; oldSettingValue = device.getSetting(it.key) // library marker kkossev.deviceProfileLib, line 1395 if (settingType == null) { logDebug "validateAndFixPreferences: settingType not found for preference ${it.key}" ; return false } // library marker kkossev.deviceProfileLib, line 1396 if (debug) { logTrace "validateAndFixPreferences: preference ${it.key} (dp=${it.value}) oldSettingValue = ${oldSettingValue} mapType = ${foundMap.type} settingType=${settingType}" } // library marker kkossev.deviceProfileLib, line 1397 if (foundMap.type != settingType) { // library marker kkossev.deviceProfileLib, line 1398 logDebug "validateAndFixPreferences: preference ${it.key} (dp=${it.value}) new mapType = ${foundMap.type} differs from the old settingType=${settingType} (oldSettingValue = ${oldSettingValue}) " // library marker kkossev.deviceProfileLib, line 1399 validationFailures ++ // library marker kkossev.deviceProfileLib, line 1400 // remove the setting and create a new one using the foundMap.type // library marker kkossev.deviceProfileLib, line 1401 try { // library marker kkossev.deviceProfileLib, line 1402 device.removeSetting(it.key) ; logDebug "validateAndFixPreferences: removing setting ${it.key}" // library marker kkossev.deviceProfileLib, line 1403 } catch (e) { // library marker kkossev.deviceProfileLib, line 1404 logWarn "validateAndFixPreferences: exception ${e} caught while removing setting ${it.key}" ; return false // library marker kkossev.deviceProfileLib, line 1405 } // library marker kkossev.deviceProfileLib, line 1406 // first, try to use the old setting value // library marker kkossev.deviceProfileLib, line 1407 try { // library marker kkossev.deviceProfileLib, line 1408 // correct the oldSettingValue type // library marker kkossev.deviceProfileLib, line 1409 if (foundMap.type == 'decimal') { newValue = oldSettingValue.toDouble() } // library marker kkossev.deviceProfileLib, line 1410 else if (foundMap.type == 'number') { newValue = oldSettingValue.toInteger() } // library marker kkossev.deviceProfileLib, line 1411 else if (foundMap.type == 'bool') { newValue = oldSettingValue == 'true' ? 1 : 0 } // library marker kkossev.deviceProfileLib, line 1412 else if (foundMap.type == 'enum') { // library marker kkossev.deviceProfileLib, line 1413 // check if the old settingValue was 'true' or 'false' and convert it to 1 or 0 // library marker kkossev.deviceProfileLib, line 1414 if (oldSettingValue == 'true' || oldSettingValue == 'false' || oldSettingValue == true || oldSettingValue == false) { // library marker kkossev.deviceProfileLib, line 1415 newValue = (oldSettingValue == 'true' || oldSettingValue == true) ? '1' : '0' // library marker kkossev.deviceProfileLib, line 1416 } // library marker kkossev.deviceProfileLib, line 1417 // check if there are any period chars in the foundMap.map string keys as String and format the settingValue as string with 2 decimals // library marker kkossev.deviceProfileLib, line 1418 else if (foundMap.map.keySet().toString().any { it.contains('.') }) { // library marker kkossev.deviceProfileLib, line 1419 newValue = String.format('%.2f', oldSettingValue) // library marker kkossev.deviceProfileLib, line 1420 } else { // library marker kkossev.deviceProfileLib, line 1421 // format the settingValue as a string of the integer value // library marker kkossev.deviceProfileLib, line 1422 newValue = String.format('%d', oldSettingValue) // library marker kkossev.deviceProfileLib, line 1423 } // library marker kkossev.deviceProfileLib, line 1424 } // library marker kkossev.deviceProfileLib, line 1425 device.updateSetting(it.key, [value:newValue, type:foundMap.type]) // library marker kkossev.deviceProfileLib, line 1426 logDebug "validateAndFixPreferences: removed and updated setting ${it.key} from old type ${settingType} to new type ${foundMap.type} with the old value ${oldSettingValue} to new value ${newValue}" // library marker kkossev.deviceProfileLib, line 1427 validationFixes ++ // library marker kkossev.deviceProfileLib, line 1428 } // library marker kkossev.deviceProfileLib, line 1429 catch (e) { // library marker kkossev.deviceProfileLib, line 1430 logWarn "validateAndFixPreferences: exception '${e}' caught while creating setting ${it.key} with type ${foundMap.type} to new type ${foundMap.type} with the old value ${oldSettingValue} to new value ${newValue}" // library marker kkossev.deviceProfileLib, line 1431 // change the settingValue to the foundMap default value // library marker kkossev.deviceProfileLib, line 1432 try { // library marker kkossev.deviceProfileLib, line 1433 settingValue = foundMap.defVal // library marker kkossev.deviceProfileLib, line 1434 device.updateSetting(it.key, [value:settingValue, type:foundMap.type]) // library marker kkossev.deviceProfileLib, line 1435 logDebug "validateAndFixPreferences: updated setting ${it.key} from old type ${settingType} to new type ${foundMap.type} with default value ${newValue} " // library marker kkossev.deviceProfileLib, line 1436 validationFixes ++ // library marker kkossev.deviceProfileLib, line 1437 } catch (e2) { // library marker kkossev.deviceProfileLib, line 1438 logWarn "validateAndFixPreferences: exception '${e2}' caught while setting default value ... Giving up!" ; return false // library marker kkossev.deviceProfileLib, line 1439 } // library marker kkossev.deviceProfileLib, line 1440 } // library marker kkossev.deviceProfileLib, line 1441 } // library marker kkossev.deviceProfileLib, line 1442 total ++ // library marker kkossev.deviceProfileLib, line 1443 } // library marker kkossev.deviceProfileLib, line 1444 logDebug "validateAndFixPreferences: total = ${total} validationFailures = ${validationFailures} validationFixes = ${validationFixes}" // library marker kkossev.deviceProfileLib, line 1445 return true // library marker kkossev.deviceProfileLib, line 1446 } // library marker kkossev.deviceProfileLib, line 1447 public String fingerprintIt(Map profileMap, Map fingerprint) { // library marker kkossev.deviceProfileLib, line 1449 if (profileMap == null) { return 'profileMap is null' } // library marker kkossev.deviceProfileLib, line 1450 if (fingerprint == null) { return 'fingerprint is null' } // library marker kkossev.deviceProfileLib, line 1451 Map defaultFingerprint = profileMap.defaultFingerprint ?: [:] // library marker kkossev.deviceProfileLib, line 1452 // if there is no defaultFingerprint, use the fingerprint as is // library marker kkossev.deviceProfileLib, line 1453 if (defaultFingerprint == [:]) { // library marker kkossev.deviceProfileLib, line 1454 return fingerprint.toString() // library marker kkossev.deviceProfileLib, line 1455 } // library marker kkossev.deviceProfileLib, line 1456 // for the missing keys, use the default values // library marker kkossev.deviceProfileLib, line 1457 String fingerprintStr = '' // library marker kkossev.deviceProfileLib, line 1458 defaultFingerprint.each { key, value -> // library marker kkossev.deviceProfileLib, line 1459 String keyValue = fingerprint[key] ?: value // library marker kkossev.deviceProfileLib, line 1460 fingerprintStr += "${key}:'${keyValue}', " // library marker kkossev.deviceProfileLib, line 1461 } // library marker kkossev.deviceProfileLib, line 1462 // remove the last comma and space // library marker kkossev.deviceProfileLib, line 1463 fingerprintStr = fingerprintStr[0..-3] // library marker kkossev.deviceProfileLib, line 1464 return fingerprintStr // library marker kkossev.deviceProfileLib, line 1465 } // library marker kkossev.deviceProfileLib, line 1466 public void printFingerprints() { // library marker kkossev.deviceProfileLib, line 1468 int count = 0 // library marker kkossev.deviceProfileLib, line 1469 deviceProfilesV3.each { profileName, profileMap -> // library marker kkossev.deviceProfileLib, line 1470 logInfo "Device Profile: ${profileName}" // library marker kkossev.deviceProfileLib, line 1471 profileMap.fingerprints?.each { fingerprint -> // library marker kkossev.deviceProfileLib, line 1472 log.info "${fingerprintIt(profileMap, fingerprint)}" // library marker kkossev.deviceProfileLib, line 1473 count++ // library marker kkossev.deviceProfileLib, line 1474 } // library marker kkossev.deviceProfileLib, line 1475 } // library marker kkossev.deviceProfileLib, line 1476 logInfo "Total fingerprints: ${count}" // library marker kkossev.deviceProfileLib, line 1477 } // library marker kkossev.deviceProfileLib, line 1478 public void printPreferences() { // library marker kkossev.deviceProfileLib, line 1480 logDebug "printPreferences: DEVICE?.preferences=${DEVICE?.preferences}" // library marker kkossev.deviceProfileLib, line 1481 if (DEVICE != null && DEVICE?.preferences != null && DEVICE?.preferences != [:] && DEVICE?.device?.isDepricated != true) { // library marker kkossev.deviceProfileLib, line 1482 (DEVICE?.preferences).each { key, value -> // library marker kkossev.deviceProfileLib, line 1483 Map inputMap = inputIt(key, true) // debug = true // library marker kkossev.deviceProfileLib, line 1484 if (inputMap != null && inputMap != [:]) { // library marker kkossev.deviceProfileLib, line 1485 log.info inputMap // library marker kkossev.deviceProfileLib, line 1486 } // library marker kkossev.deviceProfileLib, line 1487 } // library marker kkossev.deviceProfileLib, line 1488 } // library marker kkossev.deviceProfileLib, line 1489 } // library marker kkossev.deviceProfileLib, line 1490 // ~~~~~ end include (142) kkossev.deviceProfileLib ~~~~~ // ~~~~~ start include (144) kkossev.commonLib ~~~~~ /* groovylint-disable CompileStatic, DuplicateListLiteral, DuplicateMapLiteral, DuplicateNumberLiteral, DuplicateStringLiteral, ImplicitClosureParameter, ImplicitReturnStatement, InsecureRandom, LineLength, MethodCount, MethodReturnTypeRequired, MethodSize, NglParseError, NoDouble, ParameterName, PublicMethodsBeforeNonPublicMethods, StaticMethodsBeforeInstanceMethods, UnnecessaryGetter, UnnecessaryGroovyImport, UnnecessaryObjectReferences, UnnecessaryPackageReference, UnnecessaryPublicModifier, UnnecessarySetter, UnusedImport, UnusedPrivateMethod, VariableName */ // library marker kkossev.commonLib, line 1 library( // library marker kkossev.commonLib, line 2 base: 'driver', author: 'Krassimir Kossev', category: 'zigbee', description: 'Common ZCL Library', name: 'commonLib', namespace: 'kkossev', // library marker kkossev.commonLib, line 3 importUrl: 'https://raw.githubusercontent.com/kkossev/Hubitat/refs/heads/development/Libraries/commonLib.groovy', documentationLink: 'https://github.com/kkossev/Hubitat/wiki/libraries-commonLib', // library marker kkossev.commonLib, line 4 version: '3.4.0' // library marker kkossev.commonLib, line 5 ) // library marker kkossev.commonLib, line 6 /* // library marker kkossev.commonLib, line 7 * Common ZCL Library // library marker kkossev.commonLib, line 8 * // library marker kkossev.commonLib, line 9 * Licensed Virtual the Apache License, Version 2.0 (the "License"); you may not use this file except // library marker kkossev.commonLib, line 10 * in compliance with the License. You may obtain a copy of the License at: // library marker kkossev.commonLib, line 11 * // library marker kkossev.commonLib, line 12 * http://www.apache.org/licenses/LICENSE-2.0 // library marker kkossev.commonLib, line 13 * // library marker kkossev.commonLib, line 14 * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed // library marker kkossev.commonLib, line 15 * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License // library marker kkossev.commonLib, line 16 * for the specific language governing permissions and limitations under the License. // library marker kkossev.commonLib, line 17 * // library marker kkossev.commonLib, line 18 * This library is inspired by @w35l3y work on Tuya device driver (Edge project). // library marker kkossev.commonLib, line 19 * For a big portions of code all credits go to Jonathan Bradshaw. // library marker kkossev.commonLib, line 20 * // library marker kkossev.commonLib, line 21 * // library marker kkossev.commonLib, line 22 * ver. 1.0.0 2022-06-18 kkossev - first beta version // library marker kkossev.commonLib, line 23 * ver. 2.0.0 2023-05-08 kkossev - first published version 2.x.x // library marker kkossev.commonLib, line 24 * ver. 2.1.6 2023-11-06 kkossev - last update on version 2.x.x // library marker kkossev.commonLib, line 25 * ver. 3.0.0 2023-11-16 kkossev - first version 3.x.x // library marker kkossev.commonLib, line 26 * ver. 3.0.1 2023-12-06 kkossev - info event renamed to Status; txtEnable and logEnable moved to the custom driver settings; 0xFC11 cluster; logEnable is false by default; checkDriverVersion is called on updated() and on healthCheck(); // library marker kkossev.commonLib, line 27 * ver. 3.0.2 2023-12-17 kkossev - configure() changes; Groovy Lint, Format and Fix v3.0.0 // library marker kkossev.commonLib, line 28 * ver. 3.0.3 2024-03-17 kkossev - more groovy lint; support for deviceType Plug; ignore repeated temperature readings; cleaned thermostat specifics; cleaned AirQuality specifics; removed IRBlaster type; removed 'radar' type; threeStateEnable initlilization // library marker kkossev.commonLib, line 29 * ver. 3.0.4 2024-04-02 kkossev - removed Button, buttonDimmer and Fingerbot specifics; batteryVoltage bug fix; inverceSwitch bug fix; parseE002Cluster; // library marker kkossev.commonLib, line 30 * ver. 3.0.5 2024-04-05 kkossev - button methods bug fix; configure() bug fix; handlePm25Event bug fix; // library marker kkossev.commonLib, line 31 * ver. 3.0.6 2024-04-08 kkossev - removed isZigUSB() dependency; removed aqaraCube() dependency; removed button code; removed lightSensor code; moved zigbeeGroups and level and battery methods to dedicated libs + setLevel bug fix; // library marker kkossev.commonLib, line 32 * ver. 3.0.7 2024-04-23 kkossev - tuyaMagic() for Tuya devices only; added stats cfgCtr, instCtr rejoinCtr, matchDescCtr, activeEpRqCtr; trace ZDO commands; added 0x0406 OccupancyCluster; reduced debug for chatty devices; // library marker kkossev.commonLib, line 33 * ver. 3.1.0 2024-04-28 kkossev - unnecesery unschedule() speed optimization; added syncTuyaDateTime(); tuyaBlackMagic() initialization bug fix. // library marker kkossev.commonLib, line 34 * ver. 3.1.1 2024-05-05 kkossev - getTuyaAttributeValue bug fix; added customCustomParseIlluminanceCluster method // library marker kkossev.commonLib, line 35 * ver. 3.2.0 2024-05-23 kkossev - standardParse____Cluster and customParse___Cluster methods; moved onOff methods to a new library; rename all custom handlers in the libs to statdndardParseXXX // library marker kkossev.commonLib, line 36 * ver. 3.2.1 2024-06-05 kkossev - 4 in 1 V3 compatibility; added IAS cluster; setDeviceNameAndProfile() fix; // library marker kkossev.commonLib, line 37 * ver. 3.2.2 2024-06-12 kkossev - removed isAqaraTRV_OLD() and isAqaraTVOC_OLD() dependencies from the lib; added timeToHMS(); metering and electricalMeasure clusters swapped bug fix; added cluster 0x0204; // library marker kkossev.commonLib, line 38 * ver. 3.3.0 2024-06-25 kkossev - fixed exception for unknown clusters; added cluster 0xE001; added powerSource - if 5 minutes after initialize() the powerSource is still unknown, query the device for the powerSource // library marker kkossev.commonLib, line 39 * ver. 3.3.1 2024-07-06 kkossev - removed isFingerbot() dependancy; added FC03 cluster (Frient); removed noDef from the linter; added customParseIasMessage and standardParseIasMessage; powerSource set to unknown on initialize(); add 0x0007 onOffConfiguration cluster'; // library marker kkossev.commonLib, line 40 * ver. 3.3.2 2024-07-12 kkossev - added PollControl (0x0020) cluster; ping for SONOFF // library marker kkossev.commonLib, line 41 * ver. 3.3.3 2024-09-15 kkossev - added queryAllTuyaDP(); 2 minutes healthCheck option; // library marker kkossev.commonLib, line 42 * ver. 3.3.4 2025-01-29 kkossev - 'LOAD ALL DEFAULTS' is the default Configure command. // library marker kkossev.commonLib, line 43 * ver. 3.3.5 2025-03-05 kkossev - getTuyaAttributeValue made public; fixed checkDriverVersion bug on hub reboot. // library marker kkossev.commonLib, line 44 * ver. 3.4.0 2025-03-23 kkossev - healthCheck by pinging the device; updateRxStats() replaced with inline code; added state.lastRx.timeStamp; added activeEndpoints() handler call; documentation improvements // library marker kkossev.commonLib, line 45 * ver. 3.5.0 2025-04-08 kkossev - urgent fix for java.lang.CloneNotSupportedException // library marker kkossev.commonLib, line 46 * // library marker kkossev.commonLib, line 47 * TODO: add GetInfo (endpoints list) command (in the 'Tuya Device' driver?) // library marker kkossev.commonLib, line 48 * TODO: make the configure() without parameter smart - analyze the State variables and call delete states.... call ActiveAndpoints() or/amd initialize() or/and configure() // library marker kkossev.commonLib, line 49 * TODO: check - offlineCtr is not increasing? (ZBMicro); // library marker kkossev.commonLib, line 50 * TODO: check deviceCommandTimeout() // library marker kkossev.commonLib, line 51 * TODO: when device rejoins the network, read the battery percentage again (probably in custom handler, not for all devices) // library marker kkossev.commonLib, line 52 * TODO: refresh() to include updating the softwareBuild data version // library marker kkossev.commonLib, line 53 * TODO: map the ZCL powerSource options to Hubitat powerSource options // library marker kkossev.commonLib, line 54 * TODO: MOVE ZDO counters to health state? // library marker kkossev.commonLib, line 55 * TODO: refresh() to bypass the duplicated events and minimim delta time between events checks // library marker kkossev.commonLib, line 56 * TODO: Versions of the main module + included libraries (in the 'Tuya Device' driver?) // library marker kkossev.commonLib, line 57 * TODO: disableDefaultResponse for Tuya commands // library marker kkossev.commonLib, line 58 * // library marker kkossev.commonLib, line 59 */ // library marker kkossev.commonLib, line 60 String commonLibVersion() { '3.5.0' } // library marker kkossev.commonLib, line 62 String commonLibStamp() { '2025/04/08 8:36 PM' } // library marker kkossev.commonLib, line 63 import groovy.transform.Field // library marker kkossev.commonLib, line 65 import hubitat.device.HubMultiAction // library marker kkossev.commonLib, line 66 import hubitat.device.Protocol // library marker kkossev.commonLib, line 67 import hubitat.helper.HexUtils // library marker kkossev.commonLib, line 68 import hubitat.zigbee.zcl.DataType // library marker kkossev.commonLib, line 69 import java.util.concurrent.ConcurrentHashMap // library marker kkossev.commonLib, line 70 import groovy.json.JsonOutput // library marker kkossev.commonLib, line 71 import groovy.transform.CompileStatic // library marker kkossev.commonLib, line 72 import java.math.BigDecimal // library marker kkossev.commonLib, line 73 metadata { // library marker kkossev.commonLib, line 75 if (_DEBUG) { // library marker kkossev.commonLib, line 76 command 'test', [[name: 'test', type: 'STRING', description: 'test', defaultValue : '']] // library marker kkossev.commonLib, line 77 command 'testParse', [[name: 'testParse', type: 'STRING', description: 'testParse', defaultValue : '']] // library marker kkossev.commonLib, line 78 command 'tuyaTest', [ // library marker kkossev.commonLib, line 79 [name:'dpCommand', type: 'STRING', description: 'Tuya DP Command', constraints: ['STRING']], // library marker kkossev.commonLib, line 80 [name:'dpValue', type: 'STRING', description: 'Tuya DP value', constraints: ['STRING']], // library marker kkossev.commonLib, line 81 [name:'dpType', type: 'ENUM', constraints: ['DP_TYPE_VALUE', 'DP_TYPE_BOOL', 'DP_TYPE_ENUM'], description: 'DP data type'] // library marker kkossev.commonLib, line 82 ] // library marker kkossev.commonLib, line 83 } // library marker kkossev.commonLib, line 84 // common capabilities for all device types // library marker kkossev.commonLib, line 86 capability 'Configuration' // library marker kkossev.commonLib, line 87 capability 'Refresh' // library marker kkossev.commonLib, line 88 capability 'HealthCheck' // library marker kkossev.commonLib, line 89 capability 'PowerSource' // powerSource - ENUM ["battery", "dc", "mains", "unknown"] // library marker kkossev.commonLib, line 90 // common attributes for all device types // library marker kkossev.commonLib, line 92 attribute 'healthStatus', 'enum', ['unknown', 'offline', 'online'] // library marker kkossev.commonLib, line 93 attribute 'rtt', 'number' // library marker kkossev.commonLib, line 94 attribute 'Status', 'string' // library marker kkossev.commonLib, line 95 // common commands for all device types // library marker kkossev.commonLib, line 97 command 'configure', [[name:'normally it is not needed to configure anything', type: 'ENUM', constraints: ConfigureOpts.keySet() as List]] // library marker kkossev.commonLib, line 98 // trap for Hubitat F2 bug // library marker kkossev.commonLib, line 100 fingerprint profileId:'0104', endpointId:'F2', inClusters:'', outClusters:'', model:'unknown', manufacturer:'unknown', deviceJoinName: 'Zigbee device affected by Hubitat F2 bug' // library marker kkossev.commonLib, line 101 preferences { // library marker kkossev.commonLib, line 103 // txtEnable and logEnable moved to the custom driver settings - copy& paste there ... // library marker kkossev.commonLib, line 104 //input name: 'txtEnable', type: 'bool', title: 'Enable descriptionText logging', defaultValue: true, description: 'Enables command logging.' // library marker kkossev.commonLib, line 105 //input name: 'logEnable', type: 'bool', title: 'Enable debug logging', defaultValue: true, description: 'Turns on debug logging for 24 hours.' // library marker kkossev.commonLib, line 106 if (device) { // library marker kkossev.commonLib, line 108 input name: 'advancedOptions', type: 'bool', title: 'Advanced Options', description: 'These advanced options should be already automatically set in an optimal way for your device...', defaultValue: false // library marker kkossev.commonLib, line 109 if (advancedOptions == true) { // library marker kkossev.commonLib, line 110 input name: 'healthCheckMethod', type: 'enum', title: 'Healthcheck Method', options: HealthcheckMethodOpts.options, defaultValue: HealthcheckMethodOpts.defaultValue, required: true, description: 'Method to check device online/offline status.' // library marker kkossev.commonLib, line 111 input name: 'healthCheckInterval', type: 'enum', title: 'Healthcheck Interval', options: HealthcheckIntervalOpts.options, defaultValue: HealthcheckIntervalOpts.defaultValue, required: true, description: 'How often the hub will check the device health.
3 consecutive failures will result in status "offline"' // library marker kkossev.commonLib, line 112 input name: 'traceEnable', type: 'bool', title: 'Enable trace logging', defaultValue: false, description: 'Turns on detailed extra trace logging for 30 minutes.' // library marker kkossev.commonLib, line 113 } // library marker kkossev.commonLib, line 114 } // library marker kkossev.commonLib, line 115 } // library marker kkossev.commonLib, line 116 } // library marker kkossev.commonLib, line 117 @Field static final Integer DIGITAL_TIMER = 1000 // command was sent by this driver // library marker kkossev.commonLib, line 119 @Field static final Integer REFRESH_TIMER = 6000 // refresh time in miliseconds // library marker kkossev.commonLib, line 120 @Field static final Integer DEBOUNCING_TIMER = 300 // ignore switch events // library marker kkossev.commonLib, line 121 @Field static final Integer COMMAND_TIMEOUT = 10 // timeout time in seconds // library marker kkossev.commonLib, line 122 @Field static final Integer MAX_PING_MILISECONDS = 10000 // rtt more than 10 seconds will be ignored // library marker kkossev.commonLib, line 123 @Field static final String UNKNOWN = 'UNKNOWN' // library marker kkossev.commonLib, line 124 @Field static final Integer DEFAULT_MIN_REPORTING_TIME = 10 // send the report event no more often than 10 seconds by default // library marker kkossev.commonLib, line 125 @Field static final Integer DEFAULT_MAX_REPORTING_TIME = 3600 // library marker kkossev.commonLib, line 126 @Field static final Integer PRESENCE_COUNT_THRESHOLD = 3 // missing 3 checks will set the device healthStatus to offline // library marker kkossev.commonLib, line 127 @Field static final int DELAY_MS = 200 // Delay in between zigbee commands // library marker kkossev.commonLib, line 128 @Field static final Integer INFO_AUTO_CLEAR_PERIOD = 60 // automatically clear the Info attribute after 60 seconds // library marker kkossev.commonLib, line 129 @Field static final Map HealthcheckMethodOpts = [ // used by healthCheckMethod // library marker kkossev.commonLib, line 131 defaultValue: 1, options: [0: 'Disabled', 1: 'Activity check', 2: 'Periodic polling'] // library marker kkossev.commonLib, line 132 ] // library marker kkossev.commonLib, line 133 @Field static final Map HealthcheckIntervalOpts = [ // used by healthCheckInterval // library marker kkossev.commonLib, line 134 defaultValue: 240, options: [2: 'Every 2 Mins', 10: 'Every 10 Mins', 30: 'Every 30 Mins', 60: 'Every 1 Hour', 240: 'Every 4 Hours', 720: 'Every 12 Hours'] // library marker kkossev.commonLib, line 135 ] // library marker kkossev.commonLib, line 136 @Field static final Map ConfigureOpts = [ // library marker kkossev.commonLib, line 138 '*** LOAD ALL DEFAULTS ***' : [key:0, function: 'loadAllDefaults'], // library marker kkossev.commonLib, line 139 'Configure the device' : [key:2, function: 'configureNow'], // library marker kkossev.commonLib, line 140 'Reset Statistics' : [key:9, function: 'resetStatistics'], // library marker kkossev.commonLib, line 141 ' -- ' : [key:3, function: 'configureHelp'], // library marker kkossev.commonLib, line 142 'Delete All Preferences' : [key:4, function: 'deleteAllSettings'], // library marker kkossev.commonLib, line 143 'Delete All Current States' : [key:5, function: 'deleteAllCurrentStates'], // library marker kkossev.commonLib, line 144 'Delete All Scheduled Jobs' : [key:6, function: 'deleteAllScheduledJobs'], // library marker kkossev.commonLib, line 145 'Delete All State Variables' : [key:7, function: 'deleteAllStates'], // library marker kkossev.commonLib, line 146 'Delete All Child Devices' : [key:8, function: 'deleteAllChildDevices'], // library marker kkossev.commonLib, line 147 ' - ' : [key:1, function: 'configureHelp'] // library marker kkossev.commonLib, line 148 ] // library marker kkossev.commonLib, line 149 public boolean isVirtual() { device.controllerType == null || device.controllerType == '' } // library marker kkossev.commonLib, line 151 /** // library marker kkossev.commonLib, line 153 * Parse Zigbee message // library marker kkossev.commonLib, line 154 * @param description Zigbee message in hex format // library marker kkossev.commonLib, line 155 */ // library marker kkossev.commonLib, line 156 public void parse(final String description) { // library marker kkossev.commonLib, line 157 Map stateCopy = state // .clone() throws java.lang.CloneNotSupportedException in HE platform version 2.4.1.155 ! // library marker kkossev.commonLib, line 158 checkDriverVersion(stateCopy) // +1 ms // library marker kkossev.commonLib, line 159 if (state.stats != null) { state.stats?.rxCtr= (state.stats?.rxCtr ?: 0) + 1 } else { state.stats = [:] } // updateRxStats(state) // +1 ms // library marker kkossev.commonLib, line 160 if (state.lastRx != null) { state.lastRx?.timeStamp = unix2formattedDate(now()) } else { state.lastRx = [:] } // library marker kkossev.commonLib, line 161 unscheduleCommandTimeoutCheck(state) // library marker kkossev.commonLib, line 162 setHealthStatusOnline(state) // +2 ms // library marker kkossev.commonLib, line 163 if (description?.startsWith('zone status') || description?.startsWith('zone report')) { // library marker kkossev.commonLib, line 165 logDebug "parse: zone status: $description" // library marker kkossev.commonLib, line 166 if (this.respondsTo('customParseIasMessage')) { customParseIasMessage(description) } // library marker kkossev.commonLib, line 167 else if (this.respondsTo('standardParseIasMessage')) { standardParseIasMessage(description) } // library marker kkossev.commonLib, line 168 else if (this.respondsTo('parseIasMessage')) { parseIasMessage(description) } // library marker kkossev.commonLib, line 169 else { logDebug "ignored IAS zone status (no IAS parser) description: $description" } // library marker kkossev.commonLib, line 170 return // library marker kkossev.commonLib, line 171 } // library marker kkossev.commonLib, line 172 else if (description?.startsWith('enroll request')) { // library marker kkossev.commonLib, line 173 logDebug "parse: enroll request: $description" // library marker kkossev.commonLib, line 174 /* The Zone Enroll Request command is generated when a device embodying the Zone server cluster wishes to be enrolled as an active alarm device. It must do this immediately it has joined the network (during commissioning). */ // library marker kkossev.commonLib, line 175 if (settings?.logEnable) { logInfo 'Sending IAS enroll response...' } // library marker kkossev.commonLib, line 176 List cmds = zigbee.enrollResponse() + zigbee.readAttribute(0x0500, 0x0000) // library marker kkossev.commonLib, line 177 logDebug "enroll response: ${cmds}" // library marker kkossev.commonLib, line 178 sendZigbeeCommands(cmds) // library marker kkossev.commonLib, line 179 return // library marker kkossev.commonLib, line 180 } // library marker kkossev.commonLib, line 181 if (isTuyaE00xCluster(description) == true || otherTuyaOddities(description) == true) { // +15 ms // library marker kkossev.commonLib, line 183 return // library marker kkossev.commonLib, line 184 } // library marker kkossev.commonLib, line 185 final Map descMap = myParseDescriptionAsMap(description) // +5 ms // library marker kkossev.commonLib, line 186 if (!isChattyDeviceReport(descMap)) { logDebug "parse: descMap = ${descMap} description=${description }" } // library marker kkossev.commonLib, line 188 if (isSpammyDeviceReport(descMap)) { return } // +20 mS (both) // library marker kkossev.commonLib, line 189 if (descMap.profileId == '0000') { // library marker kkossev.commonLib, line 191 parseZdoClusters(descMap) // library marker kkossev.commonLib, line 192 return // library marker kkossev.commonLib, line 193 } // library marker kkossev.commonLib, line 194 if (descMap.isClusterSpecific == false) { // library marker kkossev.commonLib, line 195 parseGeneralCommandResponse(descMap) // library marker kkossev.commonLib, line 196 return // library marker kkossev.commonLib, line 197 } // library marker kkossev.commonLib, line 198 // // library marker kkossev.commonLib, line 199 if (standardAndCustomParseCluster(descMap, description)) { return } // library marker kkossev.commonLib, line 200 // // library marker kkossev.commonLib, line 201 switch (descMap.clusterInt as Integer) { // library marker kkossev.commonLib, line 202 case 0x000C : // special case : ZigUSB // Aqara TVOC Air Monitor; Aqara Cube T1 Pro; // library marker kkossev.commonLib, line 203 if (this.respondsTo('customParseAnalogInputClusterDescription')) { // library marker kkossev.commonLib, line 204 customParseAnalogInputClusterDescription(descMap, description) // ZigUSB // library marker kkossev.commonLib, line 205 descMap.remove('additionalAttrs')?.each { final Map map -> customParseAnalogInputClusterDescription(descMap + map, description) } // library marker kkossev.commonLib, line 206 } // library marker kkossev.commonLib, line 207 break // library marker kkossev.commonLib, line 208 case 0x0300 : // Patch - need refactoring of the standardParseColorControlCluster ! // library marker kkossev.commonLib, line 209 if (this.respondsTo('standardParseColorControlCluster')) { // library marker kkossev.commonLib, line 210 standardParseColorControlCluster(descMap, description) // library marker kkossev.commonLib, line 211 descMap.remove('additionalAttrs')?.each { final Map map -> standardParseColorControlCluster(descMap + map, description) } // library marker kkossev.commonLib, line 212 } // library marker kkossev.commonLib, line 213 break // library marker kkossev.commonLib, line 214 default: // library marker kkossev.commonLib, line 215 if (settings.logEnable) { // library marker kkossev.commonLib, line 216 logWarn "parse: zigbee received unknown cluster:${descMap.cluster} (${descMap.clusterInt}) message (${descMap})" // library marker kkossev.commonLib, line 217 } // library marker kkossev.commonLib, line 218 break // library marker kkossev.commonLib, line 219 } // library marker kkossev.commonLib, line 220 } // library marker kkossev.commonLib, line 221 @Field static final Map ClustersMap = [ // library marker kkossev.commonLib, line 223 0x0000: 'Basic', 0x0001: 'Power', 0x0003: 'Identify', 0x0004: 'Groups', 0x0005: 'Scenes', 0x0006: 'OnOff', 0x0007:'onOffConfiguration', 0x0008: 'LevelControl', // library marker kkossev.commonLib, line 224 0x000C: 'AnalogInput', 0x0012: 'MultistateInput', 0x0020: 'PollControl', 0x0102: 'WindowCovering', 0x0201: 'Thermostat', 0x0204: 'ThermostatConfig',/*0x0300: 'ColorControl',*/ // library marker kkossev.commonLib, line 225 0x0400: 'Illuminance', 0x0402: 'Temperature', 0x0405: 'Humidity', 0x0406: 'Occupancy', 0x042A: 'Pm25', 0x0500: 'IAS', 0x0702: 'Metering', // library marker kkossev.commonLib, line 226 0x0B04: 'ElectricalMeasure', 0xE001: 'E0001', 0xE002: 'E002', 0xEC03: 'EC03', 0xEF00: 'Tuya', 0xFC03: 'FC03', 0xFC11: 'FC11', 0xFC7E: 'AirQualityIndex', // Sensirion VOC index // library marker kkossev.commonLib, line 227 0xFCC0: 'XiaomiFCC0', // library marker kkossev.commonLib, line 228 ] // library marker kkossev.commonLib, line 229 // first try calling the custom parser, if not found, call the standard parser // library marker kkossev.commonLib, line 231 /* groovylint-disable-next-line UnusedMethodParameter */ // library marker kkossev.commonLib, line 232 boolean standardAndCustomParseCluster(Map descMap, final String description) { // library marker kkossev.commonLib, line 233 Integer clusterInt = descMap.clusterInt as Integer // library marker kkossev.commonLib, line 234 String clusterName = ClustersMap[clusterInt] ?: UNKNOWN // library marker kkossev.commonLib, line 235 if (clusterName == null || clusterName == UNKNOWN) { // library marker kkossev.commonLib, line 236 logWarn "standardAndCustomParseCluster: zigbee received unknown cluster:0x${descMap.cluster} (${descMap.clusterInt}) message (${descMap})" // library marker kkossev.commonLib, line 237 return false // library marker kkossev.commonLib, line 238 } // library marker kkossev.commonLib, line 239 String customParser = "customParse${clusterName}Cluster" // library marker kkossev.commonLib, line 240 // check if a custom parser is defined in the custom driver. If found there, the standard parser should be called within that custom parser, if needed // library marker kkossev.commonLib, line 241 if (this.respondsTo(customParser)) { // library marker kkossev.commonLib, line 242 this."${customParser}"(descMap) // library marker kkossev.commonLib, line 243 descMap.remove('additionalAttrs')?.each { final Map map -> this."${customParser}"(descMap + map) } // library marker kkossev.commonLib, line 244 return true // library marker kkossev.commonLib, line 245 } // library marker kkossev.commonLib, line 246 String standardParser = "standardParse${clusterName}Cluster" // library marker kkossev.commonLib, line 247 // if no custom parser is defined, try the standard parser (if exists), eventually defined in the included library file // library marker kkossev.commonLib, line 248 if (this.respondsTo(standardParser)) { // library marker kkossev.commonLib, line 249 this."${standardParser}"(descMap) // library marker kkossev.commonLib, line 250 descMap.remove('additionalAttrs')?.each { final Map map -> this."${standardParser}"(descMap + map) } // library marker kkossev.commonLib, line 251 return true // library marker kkossev.commonLib, line 252 } // library marker kkossev.commonLib, line 253 if (device?.getDataValue('model') != 'ZigUSB' && descMap.cluster != '0300') { // patch! // library marker kkossev.commonLib, line 254 logWarn "standardAndCustomParseCluster: Missing ${standardParser} or ${customParser} handler for cluster:0x${descMap.cluster} (${descMap.clusterInt}) message (${descMap})" // library marker kkossev.commonLib, line 255 } // library marker kkossev.commonLib, line 256 return false // library marker kkossev.commonLib, line 257 } // library marker kkossev.commonLib, line 258 // not used - throws exception : error groovy.lang.MissingPropertyException: No such property: rxCtr for class: java.lang.String on line 1568 (method parse) // library marker kkossev.commonLib, line 260 private static void updateRxStats(final Map state) { // library marker kkossev.commonLib, line 261 if (state.stats != null) { state.stats['rxCtr'] = (state.stats['rxCtr'] ?: 0) + 1 } else { state.stats = [:] } // +5ms // library marker kkossev.commonLib, line 262 } // library marker kkossev.commonLib, line 263 public boolean isChattyDeviceReport(final Map descMap) { // when @CompileStatis is slower? // library marker kkossev.commonLib, line 265 if (_TRACE_ALL == true) { return false } // library marker kkossev.commonLib, line 266 if (this.respondsTo('isSpammyDPsToNotTrace')) { // defined in deviceProfileLib // library marker kkossev.commonLib, line 267 return isSpammyDPsToNotTrace(descMap) // library marker kkossev.commonLib, line 268 } // library marker kkossev.commonLib, line 269 return false // library marker kkossev.commonLib, line 270 } // library marker kkossev.commonLib, line 271 public boolean isSpammyDeviceReport(final Map descMap) { // library marker kkossev.commonLib, line 273 if (_TRACE_ALL == true) { return false } // library marker kkossev.commonLib, line 274 if (this.respondsTo('isSpammyDPsToIgnore')) { // defined in deviceProfileLib // library marker kkossev.commonLib, line 275 return isSpammyDPsToIgnore(descMap) // library marker kkossev.commonLib, line 276 } // library marker kkossev.commonLib, line 277 return false // library marker kkossev.commonLib, line 278 } // library marker kkossev.commonLib, line 279 // not used? // library marker kkossev.commonLib, line 281 public boolean isSpammyTuyaRadar() { // library marker kkossev.commonLib, line 282 if (_TRACE_ALL == true) { return false } // library marker kkossev.commonLib, line 283 if (this.respondsTo('isSpammyDeviceProfile'())) { // defined in deviceProfileLib // library marker kkossev.commonLib, line 284 return isSpammyDeviceProfile() // library marker kkossev.commonLib, line 285 } // library marker kkossev.commonLib, line 286 return false // library marker kkossev.commonLib, line 287 } // library marker kkossev.commonLib, line 288 @Field static final Map ZdoClusterEnum = [ // library marker kkossev.commonLib, line 290 0x0002: 'Node Descriptor Request', 0x0005: 'Active Endpoints Request', 0x0006: 'Match Descriptor Request', 0x0022: 'Unbind Request', 0x0013: 'Device announce', 0x0034: 'Management Leave Request', // library marker kkossev.commonLib, line 291 0x8002: 'Node Descriptor Response', 0x8004: 'Simple Descriptor Response', 0x8005: 'Active Endpoints Response', 0x801D: 'Extended Simple Descriptor Response', 0x801E: 'Extended Active Endpoint Response', // library marker kkossev.commonLib, line 292 0x8021: 'Bind Response', 0x8022: 'Unbind Response', 0x8023: 'Bind Register Response', 0x8034: 'Management Leave Response' // library marker kkossev.commonLib, line 293 ] // library marker kkossev.commonLib, line 294 // ZDO (Zigbee Data Object) Clusters Parsing // library marker kkossev.commonLib, line 296 private void parseZdoClusters(final Map descMap) { // library marker kkossev.commonLib, line 297 if (state.stats == null) { state.stats = [:] } // library marker kkossev.commonLib, line 298 final Integer clusterId = descMap.clusterInt as Integer // library marker kkossev.commonLib, line 299 final String clusterName = ZdoClusterEnum[clusterId] ?: "UNKNOWN_CLUSTER (0x${descMap.clusterId})" // library marker kkossev.commonLib, line 300 final String statusHex = ((List)descMap.data)[1] // library marker kkossev.commonLib, line 301 final Integer statusCode = hexStrToUnsignedInt(statusHex) // library marker kkossev.commonLib, line 302 final String statusName = ZigbeeStatusEnum[statusCode] ?: "0x${statusHex}" // library marker kkossev.commonLib, line 303 final String clusterInfo = "${device.displayName} Received ZDO ${clusterName} (0x${descMap.clusterId}) status ${statusName}" // library marker kkossev.commonLib, line 304 List cmds = [] // library marker kkossev.commonLib, line 305 switch (clusterId) { // library marker kkossev.commonLib, line 306 case 0x0005 : // library marker kkossev.commonLib, line 307 state.stats['activeEpRqCtr'] = (state.stats['activeEpRqCtr'] ?: 0) + 1 // library marker kkossev.commonLib, line 308 if (settings?.logEnable) { log.debug "${clusterInfo}, data=${descMap.data} (Sequence Number:${descMap.data[0]}, data:${descMap.data})" } // library marker kkossev.commonLib, line 309 // send the active endpoint response // library marker kkossev.commonLib, line 310 cmds += ["he raw ${device.deviceNetworkId} 0 0 0x8005 {00 00 00 00 01 01} {0x0000}"] // library marker kkossev.commonLib, line 311 sendZigbeeCommands(cmds) // library marker kkossev.commonLib, line 312 break // library marker kkossev.commonLib, line 313 case 0x0006 : // library marker kkossev.commonLib, line 314 state.stats['matchDescCtr'] = (state.stats['matchDescCtr'] ?: 0) + 1 // library marker kkossev.commonLib, line 315 if (settings?.logEnable) { log.debug "${clusterInfo}, data=${descMap.data} (Sequence Number:${descMap.data[0]}, Input cluster count:${descMap.data[5]} Input cluster: 0x${descMap.data[7] + descMap.data[6]})" } // library marker kkossev.commonLib, line 316 cmds += ["he raw ${device.deviceNetworkId} 0 0 0x8006 {00 00 00 00 00} {0x0000}"] // library marker kkossev.commonLib, line 317 sendZigbeeCommands(cmds) // library marker kkossev.commonLib, line 318 break // library marker kkossev.commonLib, line 319 case 0x0013 : // device announcement // library marker kkossev.commonLib, line 320 state.stats['rejoinCtr'] = (state.stats['rejoinCtr'] ?: 0) + 1 // library marker kkossev.commonLib, line 321 if (settings?.logEnable) { log.debug "${clusterInfo}, rejoinCtr= ${state.stats['rejoinCtr']}, data=${descMap.data} (Sequence Number:${descMap.data[0]}, Device network ID: ${descMap.data[2] + descMap.data[1]}, Capability Information: ${descMap.data[11]})" } // library marker kkossev.commonLib, line 322 break // library marker kkossev.commonLib, line 323 case 0x8004 : // simple descriptor response // library marker kkossev.commonLib, line 324 if (settings?.logEnable) { log.debug "${clusterInfo}, data=${descMap.data} (Sequence Number:${descMap.data[0]}, status:${descMap.data[1]}, lenght:${hubitat.helper.HexUtils.hexStringToInt(descMap.data[4])}" } // library marker kkossev.commonLib, line 325 if (this.respondsTo('parseSimpleDescriptorResponse')) { parseSimpleDescriptorResponse(descMap) } // library marker kkossev.commonLib, line 326 break // library marker kkossev.commonLib, line 327 case 0x8005 : // endpoint response // library marker kkossev.commonLib, line 328 String endpointCount = descMap.data[4] // library marker kkossev.commonLib, line 329 String endpointList = descMap.data[5] // library marker kkossev.commonLib, line 330 if (settings?.logEnable) { log.debug "${clusterInfo}, (endpoint response) endpointCount = ${endpointCount} endpointList = ${endpointList}" } // library marker kkossev.commonLib, line 331 break // library marker kkossev.commonLib, line 332 case 0x8021 : // bind response // library marker kkossev.commonLib, line 333 if (settings?.logEnable) { log.debug "${clusterInfo}, data=${descMap.data} (Sequence Number:${descMap.data[0]}, Status: ${descMap.data[1] == '00' ? 'Success' : 'Failure'})" } // library marker kkossev.commonLib, line 334 break // library marker kkossev.commonLib, line 335 case 0x0002 : // Node Descriptor Request // library marker kkossev.commonLib, line 336 case 0x0036 : // Permit Joining Request // library marker kkossev.commonLib, line 337 case 0x8022 : // unbind request // library marker kkossev.commonLib, line 338 case 0x8034 : // leave response // library marker kkossev.commonLib, line 339 if (settings?.logEnable) { log.debug "${device.displayName} Unprocessed ZDO command: cluster=${descMap.clusterId} command=${descMap.command} attrId=${descMap.attrId} value=${descMap.value} data=${descMap.data}" } // library marker kkossev.commonLib, line 340 break // library marker kkossev.commonLib, line 341 default : // library marker kkossev.commonLib, line 342 if (settings?.logEnable) { log.warn "${device.displayName} Unprocessed ZDO command: cluster=${descMap.clusterId} command=${descMap.command} attrId=${descMap.attrId} value=${descMap.value} data=${descMap.data}" } // library marker kkossev.commonLib, line 343 break // library marker kkossev.commonLib, line 344 } // library marker kkossev.commonLib, line 345 if (this.respondsTo('customParseZdoClusters')) { customParseZdoClusters(descMap) } // library marker kkossev.commonLib, line 346 } // library marker kkossev.commonLib, line 347 // Zigbee General Command Parsing // library marker kkossev.commonLib, line 349 private void parseGeneralCommandResponse(final Map descMap) { // library marker kkossev.commonLib, line 350 final int commandId = hexStrToUnsignedInt(descMap.command) // library marker kkossev.commonLib, line 351 switch (commandId) { // library marker kkossev.commonLib, line 352 case 0x01: parseReadAttributeResponse(descMap); break // library marker kkossev.commonLib, line 353 case 0x04: parseWriteAttributeResponse(descMap); break // library marker kkossev.commonLib, line 354 case 0x07: parseConfigureResponse(descMap); break // library marker kkossev.commonLib, line 355 case 0x09: parseReadReportingConfigResponse(descMap); break // library marker kkossev.commonLib, line 356 case 0x0B: parseDefaultCommandResponse(descMap); break // library marker kkossev.commonLib, line 357 default: // library marker kkossev.commonLib, line 358 final String commandName = ZigbeeGeneralCommandEnum[commandId] ?: "UNKNOWN_COMMAND (0x${descMap.command})" // library marker kkossev.commonLib, line 359 final String clusterName = clusterLookup(descMap.clusterInt) // library marker kkossev.commonLib, line 360 final String status = descMap.data in List ? ((List)descMap.data).last() : descMap.data // library marker kkossev.commonLib, line 361 final int statusCode = hexStrToUnsignedInt(status) // library marker kkossev.commonLib, line 362 final String statusName = ZigbeeStatusEnum[statusCode] ?: "0x${status}" // library marker kkossev.commonLib, line 363 if (statusCode > 0x00) { // library marker kkossev.commonLib, line 364 log.warn "zigbee ${commandName} ${clusterName} error: ${statusName}" // library marker kkossev.commonLib, line 365 } else if (settings.logEnable) { // library marker kkossev.commonLib, line 366 log.trace "zigbee ${commandName} ${clusterName}: ${descMap.data}" // library marker kkossev.commonLib, line 367 } // library marker kkossev.commonLib, line 368 break // library marker kkossev.commonLib, line 369 } // library marker kkossev.commonLib, line 370 } // library marker kkossev.commonLib, line 371 // Zigbee Read Attribute Response Parsing // library marker kkossev.commonLib, line 373 private void parseReadAttributeResponse(final Map descMap) { // library marker kkossev.commonLib, line 374 final List data = descMap.data as List // library marker kkossev.commonLib, line 375 final String attribute = data[1] + data[0] // library marker kkossev.commonLib, line 376 final int statusCode = hexStrToUnsignedInt(data[2]) // library marker kkossev.commonLib, line 377 final String status = ZigbeeStatusEnum[statusCode] ?: "0x${data}" // library marker kkossev.commonLib, line 378 if (statusCode > 0x00) { // library marker kkossev.commonLib, line 379 logWarn "zigbee read ${clusterLookup(descMap.clusterInt)} attribute 0x${attribute} error: ${status}" // library marker kkossev.commonLib, line 380 } // library marker kkossev.commonLib, line 381 else { // library marker kkossev.commonLib, line 382 logDebug "zigbee read ${clusterLookup(descMap.clusterInt)} attribute 0x${attribute} response: ${status} ${data}" // library marker kkossev.commonLib, line 383 } // library marker kkossev.commonLib, line 384 } // library marker kkossev.commonLib, line 385 // Zigbee Write Attribute Response Parsing // library marker kkossev.commonLib, line 387 private void parseWriteAttributeResponse(final Map descMap) { // library marker kkossev.commonLib, line 388 final String data = descMap.data in List ? ((List)descMap.data).first() : descMap.data // library marker kkossev.commonLib, line 389 final int statusCode = hexStrToUnsignedInt(data) // library marker kkossev.commonLib, line 390 final String statusName = ZigbeeStatusEnum[statusCode] ?: "0x${data}" // library marker kkossev.commonLib, line 391 if (statusCode > 0x00) { // library marker kkossev.commonLib, line 392 logWarn "zigbee response write ${clusterLookup(descMap.clusterInt)} attribute error: ${statusName}" // library marker kkossev.commonLib, line 393 } // library marker kkossev.commonLib, line 394 else { // library marker kkossev.commonLib, line 395 logDebug "zigbee response write ${clusterLookup(descMap.clusterInt)} attribute response: ${statusName}" // library marker kkossev.commonLib, line 396 } // library marker kkossev.commonLib, line 397 } // library marker kkossev.commonLib, line 398 // Zigbee Configure Reporting Response Parsing - command 0x07 // library marker kkossev.commonLib, line 400 private void parseConfigureResponse(final Map descMap) { // library marker kkossev.commonLib, line 401 // TODO - parse the details of the configuration respose - cluster, min, max, delta ... // library marker kkossev.commonLib, line 402 final String status = ((List)descMap.data).first() // library marker kkossev.commonLib, line 403 final int statusCode = hexStrToUnsignedInt(status) // library marker kkossev.commonLib, line 404 if (statusCode == 0x00 && settings.enableReporting != false) { // library marker kkossev.commonLib, line 405 state.reportingEnabled = true // library marker kkossev.commonLib, line 406 } // library marker kkossev.commonLib, line 407 final String statusName = ZigbeeStatusEnum[statusCode] ?: "0x${status}" // library marker kkossev.commonLib, line 408 if (statusCode > 0x00) { // library marker kkossev.commonLib, line 409 log.warn "zigbee configure reporting error: ${statusName} ${descMap.data}" // library marker kkossev.commonLib, line 410 } else { // library marker kkossev.commonLib, line 411 logDebug "zigbee configure reporting response: ${statusName} ${descMap.data}" // library marker kkossev.commonLib, line 412 } // library marker kkossev.commonLib, line 413 } // library marker kkossev.commonLib, line 414 // Parses the response of reading reporting configuration - command 0x09 // library marker kkossev.commonLib, line 416 private void parseReadReportingConfigResponse(final Map descMap) { // library marker kkossev.commonLib, line 417 int status = zigbee.convertHexToInt(descMap.data[0]) // Status: Success (0x00) // library marker kkossev.commonLib, line 418 //def attr = zigbee.convertHexToInt(descMap.data[3])*256 + zigbee.convertHexToInt(descMap.data[2]) // Attribute: OnOff (0x0000) // library marker kkossev.commonLib, line 419 if (status == 0) { // library marker kkossev.commonLib, line 420 //def dataType = zigbee.convertHexToInt(descMap.data[4]) // Data Type: Boolean (0x10) // library marker kkossev.commonLib, line 421 int min = zigbee.convertHexToInt(descMap.data[6]) * 256 + zigbee.convertHexToInt(descMap.data[5]) // library marker kkossev.commonLib, line 422 int max = zigbee.convertHexToInt(descMap.data[8] + descMap.data[7]) // library marker kkossev.commonLib, line 423 int delta = 0 // library marker kkossev.commonLib, line 424 if (descMap.data.size() >= 10) { // library marker kkossev.commonLib, line 425 delta = zigbee.convertHexToInt(descMap.data[10] + descMap.data[9]) // library marker kkossev.commonLib, line 426 } // library marker kkossev.commonLib, line 427 else { // library marker kkossev.commonLib, line 428 logTrace "descMap.data.size = ${descMap.data.size()}" // library marker kkossev.commonLib, line 429 } // library marker kkossev.commonLib, line 430 logDebug "Received Read Reporting Configuration Response (0x09) for cluster:${descMap.clusterId} attribute:${descMap.data[3] + descMap.data[2]}, data=${descMap.data} (Status: ${descMap.data[0] == '00' ? 'Success' : 'Failure'}) min=${min} max=${max} delta=${delta}" // library marker kkossev.commonLib, line 431 } // library marker kkossev.commonLib, line 432 else { // library marker kkossev.commonLib, line 433 logWarn "Not Found (0x8b) Read Reporting Configuration Response for cluster:${descMap.clusterId} attribute:${descMap.data[3] + descMap.data[2]}, data=${descMap.data} (Status: ${descMap.data[0] == '00' ? 'Success' : 'Failure'})" // library marker kkossev.commonLib, line 434 } // library marker kkossev.commonLib, line 435 } // library marker kkossev.commonLib, line 436 private Boolean executeCustomHandler(String handlerName, Object handlerArgs) { // library marker kkossev.commonLib, line 438 if (!this.respondsTo(handlerName)) { // library marker kkossev.commonLib, line 439 logTrace "executeCustomHandler: function ${handlerName} not found" // library marker kkossev.commonLib, line 440 return false // library marker kkossev.commonLib, line 441 } // library marker kkossev.commonLib, line 442 // execute the customHandler function // library marker kkossev.commonLib, line 443 Boolean result = false // library marker kkossev.commonLib, line 444 try { // library marker kkossev.commonLib, line 445 result = "$handlerName"(handlerArgs) // library marker kkossev.commonLib, line 446 } // library marker kkossev.commonLib, line 447 catch (e) { // library marker kkossev.commonLib, line 448 logWarn "executeCustomHandler: Exception '${e}'caught while processing $handlerName($handlerArgs) (val=${fncmd}))" // library marker kkossev.commonLib, line 449 return false // library marker kkossev.commonLib, line 450 } // library marker kkossev.commonLib, line 451 //logDebug "customSetFunction result is ${fncmd}" // library marker kkossev.commonLib, line 452 return result // library marker kkossev.commonLib, line 453 } // library marker kkossev.commonLib, line 454 // Zigbee Default Command Response Parsing // library marker kkossev.commonLib, line 456 private void parseDefaultCommandResponse(final Map descMap) { // library marker kkossev.commonLib, line 457 final List data = descMap.data as List // library marker kkossev.commonLib, line 458 final String commandId = data[0] // library marker kkossev.commonLib, line 459 final int statusCode = hexStrToUnsignedInt(data[1]) // library marker kkossev.commonLib, line 460 final String status = ZigbeeStatusEnum[statusCode] ?: "0x${data[1]}" // library marker kkossev.commonLib, line 461 if (statusCode > 0x00) { // library marker kkossev.commonLib, line 462 logWarn "zigbee ${clusterLookup(descMap.clusterInt)} command 0x${commandId} error: ${status}" // library marker kkossev.commonLib, line 463 } else { // library marker kkossev.commonLib, line 464 logDebug "zigbee ${clusterLookup(descMap.clusterInt)} command 0x${commandId} response: ${status}" // library marker kkossev.commonLib, line 465 // ZigUSB has its own interpretation of the Zigbee standards ... :( // library marker kkossev.commonLib, line 466 if (this.respondsTo('customParseDefaultCommandResponse')) { // library marker kkossev.commonLib, line 467 customParseDefaultCommandResponse(descMap) // library marker kkossev.commonLib, line 468 } // library marker kkossev.commonLib, line 469 } // library marker kkossev.commonLib, line 470 } // library marker kkossev.commonLib, line 471 // Zigbee Attribute IDs // library marker kkossev.commonLib, line 473 @Field static final int ATTRIBUTE_READING_INFO_SET = 0x0000 // library marker kkossev.commonLib, line 474 @Field static final int FIRMWARE_VERSION_ID = 0x4000 // library marker kkossev.commonLib, line 475 @Field static final int PING_ATTR_ID = 0x01 // library marker kkossev.commonLib, line 476 @Field static final Map ZigbeeStatusEnum = [ // library marker kkossev.commonLib, line 478 0x00: 'Success', 0x01: 'Failure', 0x02: 'Not Authorized', 0x80: 'Malformed Command', 0x81: 'Unsupported COMMAND', 0x85: 'Invalid Field', 0x86: 'Unsupported Attribute', 0x87: 'Invalid Value', 0x88: 'Read Only', // library marker kkossev.commonLib, line 479 0x89: 'Insufficient Space', 0x8A: 'Duplicate Exists', 0x8B: 'Not Found', 0x8C: 'Unreportable Attribute', 0x8D: 'Invalid Data Type', 0x8E: 'Invalid Selector', 0x94: 'Time out', 0x9A: 'Notification Pending', 0xC3: 'Unsupported Cluster' // library marker kkossev.commonLib, line 480 ] // library marker kkossev.commonLib, line 481 @Field static final Map ZigbeeGeneralCommandEnum = [ // library marker kkossev.commonLib, line 483 0x00: 'Read Attributes', 0x01: 'Read Attributes Response', 0x02: 'Write Attributes', 0x03: 'Write Attributes Undivided', 0x04: 'Write Attributes Response', 0x05: 'Write Attributes No Response', 0x06: 'Configure Reporting', // library marker kkossev.commonLib, line 484 0x07: 'Configure Reporting Response', 0x08: 'Read Reporting Configuration', 0x09: 'Read Reporting Configuration Response', 0x0A: 'Report Attributes', 0x0B: 'Default Response', 0x0C: 'Discover Attributes', 0x0D: 'Discover Attributes Response', // library marker kkossev.commonLib, line 485 0x0E: 'Read Attributes Structured', 0x0F: 'Write Attributes Structured', 0x10: 'Write Attributes Structured Response', 0x11: 'Discover Commands Received', 0x12: 'Discover Commands Received Response', 0x13: 'Discover Commands Generated', // library marker kkossev.commonLib, line 486 0x14: 'Discover Commands Generated Response', 0x15: 'Discover Attributes Extended', 0x16: 'Discover Attributes Extended Response' // library marker kkossev.commonLib, line 487 ] // library marker kkossev.commonLib, line 488 @Field static final int ROLLING_AVERAGE_N = 10 // library marker kkossev.commonLib, line 490 private BigDecimal approxRollingAverage(BigDecimal avgPar, BigDecimal newSample) { // library marker kkossev.commonLib, line 491 BigDecimal avg = avgPar // library marker kkossev.commonLib, line 492 if (avg == null || avg == 0) { avg = newSample } // library marker kkossev.commonLib, line 493 avg -= avg / ROLLING_AVERAGE_N // library marker kkossev.commonLib, line 494 avg += newSample / ROLLING_AVERAGE_N // library marker kkossev.commonLib, line 495 return avg // library marker kkossev.commonLib, line 496 } // library marker kkossev.commonLib, line 497 private void handlePingResponse() { // library marker kkossev.commonLib, line 499 Long now = new Date().getTime() // library marker kkossev.commonLib, line 500 if (state.lastRx == null) { state.lastRx = [:] } // library marker kkossev.commonLib, line 501 state.lastRx['checkInTime'] = now // library marker kkossev.commonLib, line 502 int timeRunning = now.toInteger() - (state.lastTx['pingTime'] ?: '0').toInteger() // library marker kkossev.commonLib, line 504 if (timeRunning > 0 && timeRunning < MAX_PING_MILISECONDS) { // library marker kkossev.commonLib, line 505 state.stats['pingsOK'] = (state.stats['pingsOK'] ?: 0) + 1 // library marker kkossev.commonLib, line 506 if (timeRunning < safeToInt((state.stats['pingsMin'] ?: '999'))) { state.stats['pingsMin'] = timeRunning } // library marker kkossev.commonLib, line 507 if (timeRunning > safeToInt((state.stats['pingsMax'] ?: '0'))) { state.stats['pingsMax'] = timeRunning } // library marker kkossev.commonLib, line 508 state.stats['pingsAvg'] = approxRollingAverage(safeToDouble(state.stats['pingsAvg']), safeToDouble(timeRunning)) as int // library marker kkossev.commonLib, line 509 sendRttEvent() // library marker kkossev.commonLib, line 510 } // library marker kkossev.commonLib, line 511 else { // library marker kkossev.commonLib, line 512 logWarn "unexpected ping timeRunning=${timeRunning} " // library marker kkossev.commonLib, line 513 } // library marker kkossev.commonLib, line 514 state.states['isPing'] = false // library marker kkossev.commonLib, line 515 } // library marker kkossev.commonLib, line 516 /* // library marker kkossev.commonLib, line 518 * ----------------------------------------------------------------------------- // library marker kkossev.commonLib, line 519 * Standard clusters reporting handlers // library marker kkossev.commonLib, line 520 * ----------------------------------------------------------------------------- // library marker kkossev.commonLib, line 521 */ // library marker kkossev.commonLib, line 522 @Field static final Map powerSourceOpts = [ defaultValue: 0, options: [0: 'unknown', 1: 'mains', 2: 'mains', 3: 'battery', 4: 'dc', 5: 'emergency mains', 6: 'emergency mains']] // library marker kkossev.commonLib, line 523 // Zigbee Basic Cluster Parsing 0x0000 - called from the main parse method // library marker kkossev.commonLib, line 525 private void standardParseBasicCluster(final Map descMap) { // library marker kkossev.commonLib, line 526 Long now = new Date().getTime() // library marker kkossev.commonLib, line 527 if (state.lastRx == null) { state.lastRx = [:] } // library marker kkossev.commonLib, line 528 state.lastRx['checkInTime'] = now // library marker kkossev.commonLib, line 529 boolean isPing = state.states?.isPing ?: false // library marker kkossev.commonLib, line 530 switch (descMap.attrInt as Integer) { // library marker kkossev.commonLib, line 531 case 0x0000: // library marker kkossev.commonLib, line 532 logDebug "Basic cluster: ZCLVersion = ${descMap?.value}" // library marker kkossev.commonLib, line 533 break // library marker kkossev.commonLib, line 534 case PING_ATTR_ID: // 0x01 - Using 0x01 read as a simple ping/pong mechanism // library marker kkossev.commonLib, line 535 if (isPing) { // library marker kkossev.commonLib, line 536 handlePingResponse() // library marker kkossev.commonLib, line 537 } // library marker kkossev.commonLib, line 538 else { // library marker kkossev.commonLib, line 539 logTrace "Tuya check-in message (attribute ${descMap.attrId} reported: ${descMap.value})" // library marker kkossev.commonLib, line 540 } // library marker kkossev.commonLib, line 541 break // library marker kkossev.commonLib, line 542 case 0x0004: // library marker kkossev.commonLib, line 543 logDebug "received device manufacturer ${descMap?.value}" // library marker kkossev.commonLib, line 544 // received device manufacturer IKEA of Sweden // library marker kkossev.commonLib, line 545 String manufacturer = device.getDataValue('manufacturer') // library marker kkossev.commonLib, line 546 if ((manufacturer == null || manufacturer == 'unknown') && (descMap?.value != null)) { // library marker kkossev.commonLib, line 547 logWarn "updating device manufacturer from ${manufacturer} to ${descMap?.value}" // library marker kkossev.commonLib, line 548 device.updateDataValue('manufacturer', descMap?.value) // library marker kkossev.commonLib, line 549 } // library marker kkossev.commonLib, line 550 break // library marker kkossev.commonLib, line 551 case 0x0005: // library marker kkossev.commonLib, line 552 if (isPing) { // library marker kkossev.commonLib, line 553 handlePingResponse() // library marker kkossev.commonLib, line 554 } // library marker kkossev.commonLib, line 555 else { // library marker kkossev.commonLib, line 556 logDebug "received device model ${descMap?.value}" // library marker kkossev.commonLib, line 557 // received device model Remote Control N2 // library marker kkossev.commonLib, line 558 String model = device.getDataValue('model') // library marker kkossev.commonLib, line 559 if ((model == null || model == 'unknown') && (descMap?.value != null)) { // library marker kkossev.commonLib, line 560 logWarn "updating device model from ${model} to ${descMap?.value}" // library marker kkossev.commonLib, line 561 device.updateDataValue('model', descMap?.value) // library marker kkossev.commonLib, line 562 } // library marker kkossev.commonLib, line 563 } // library marker kkossev.commonLib, line 564 break // library marker kkossev.commonLib, line 565 case 0x0007: // library marker kkossev.commonLib, line 566 String powerSourceReported = powerSourceOpts.options[descMap?.value as int] // library marker kkossev.commonLib, line 567 logDebug "received Power source ${powerSourceReported} (${descMap?.value})" // library marker kkossev.commonLib, line 568 String currentPowerSource = device.getDataValue('powerSource') // library marker kkossev.commonLib, line 569 if (currentPowerSource == null || currentPowerSource == 'unknown') { // library marker kkossev.commonLib, line 570 logInfo "updating device powerSource from ${currentPowerSource} to ${powerSourceReported}" // library marker kkossev.commonLib, line 571 sendEvent(name: 'powerSource', value: powerSourceReported, type: 'physical') // library marker kkossev.commonLib, line 572 } // library marker kkossev.commonLib, line 573 break // library marker kkossev.commonLib, line 574 case 0xFFDF: // library marker kkossev.commonLib, line 575 logDebug "Tuya check-in (Cluster Revision=${descMap?.value})" // library marker kkossev.commonLib, line 576 break // library marker kkossev.commonLib, line 577 case 0xFFE2: // library marker kkossev.commonLib, line 578 logDebug "Tuya check-in (AppVersion=${descMap?.value})" // library marker kkossev.commonLib, line 579 break // library marker kkossev.commonLib, line 580 case [0xFFE0, 0xFFE1, 0xFFE3, 0xFFE4] : // library marker kkossev.commonLib, line 581 logTrace "Tuya attribute ${descMap?.attrId} value=${descMap?.value}" // library marker kkossev.commonLib, line 582 break // library marker kkossev.commonLib, line 583 case 0xFFFE: // library marker kkossev.commonLib, line 584 logTrace "Tuya attributeReportingStatus (attribute FFFE) value=${descMap?.value}" // library marker kkossev.commonLib, line 585 break // library marker kkossev.commonLib, line 586 case FIRMWARE_VERSION_ID: // 0x4000 // library marker kkossev.commonLib, line 587 final String version = descMap.value ?: 'unknown' // library marker kkossev.commonLib, line 588 log.info "device firmware version is ${version}" // library marker kkossev.commonLib, line 589 updateDataValue('softwareBuild', version) // library marker kkossev.commonLib, line 590 break // library marker kkossev.commonLib, line 591 default: // library marker kkossev.commonLib, line 592 logWarn "zigbee received unknown Basic cluster attribute 0x${descMap.attrId} (value ${descMap.value})" // library marker kkossev.commonLib, line 593 break // library marker kkossev.commonLib, line 594 } // library marker kkossev.commonLib, line 595 } // library marker kkossev.commonLib, line 596 private void standardParsePollControlCluster(final Map descMap) { // library marker kkossev.commonLib, line 598 switch (descMap.attrInt as Integer) { // library marker kkossev.commonLib, line 599 case 0x0000: logDebug "PollControl cluster: CheckInInterval = ${descMap?.value}" ; break // library marker kkossev.commonLib, line 600 case 0x0001: logDebug "PollControl cluster: LongPollInterval = ${descMap?.value}" ; break // library marker kkossev.commonLib, line 601 case 0x0002: logDebug "PollControl cluster: ShortPollInterval = ${descMap?.value}" ; break // library marker kkossev.commonLib, line 602 case 0x0003: logDebug "PollControl cluster: FastPollTimeout = ${descMap?.value}" ; break // library marker kkossev.commonLib, line 603 case 0x0004: logDebug "PollControl cluster: CheckInIntervalMin = ${descMap?.value}" ; break // library marker kkossev.commonLib, line 604 case 0x0005: logDebug "PollControl cluster: LongPollIntervalMin = ${descMap?.value}" ; break // library marker kkossev.commonLib, line 605 case 0x0006: logDebug "PollControl cluster: FastPollTimeoutMax = ${descMap?.value}" ; break // library marker kkossev.commonLib, line 606 default: logWarn "zigbee received unknown PollControl cluster attribute 0x${descMap.attrId} (value ${descMap.value})" ; break // library marker kkossev.commonLib, line 607 } // library marker kkossev.commonLib, line 608 } // library marker kkossev.commonLib, line 609 public void clearIsDigital() { state.states['isDigital'] = false } // library marker kkossev.commonLib, line 611 void switchDebouncingClear() { state.states['debounce'] = false } // library marker kkossev.commonLib, line 612 void isRefreshRequestClear() { state.states['isRefresh'] = false } // library marker kkossev.commonLib, line 613 Map myParseDescriptionAsMap(String description) { // library marker kkossev.commonLib, line 615 Map descMap = [:] // library marker kkossev.commonLib, line 616 try { // library marker kkossev.commonLib, line 617 descMap = zigbee.parseDescriptionAsMap(description) // library marker kkossev.commonLib, line 618 } // library marker kkossev.commonLib, line 619 catch (e1) { // library marker kkossev.commonLib, line 620 logWarn "exception ${e1} caught while parseDescriptionAsMap myParseDescriptionAsMap description: ${description}" // library marker kkossev.commonLib, line 621 // try alternative custom parsing // library marker kkossev.commonLib, line 622 descMap = [:] // library marker kkossev.commonLib, line 623 try { // library marker kkossev.commonLib, line 624 descMap += description.replaceAll('\\[|\\]', '').split(',').collectEntries { entry -> // library marker kkossev.commonLib, line 625 List pair = entry.split(':') // library marker kkossev.commonLib, line 626 [(pair.first().trim()): pair.last().trim()] // library marker kkossev.commonLib, line 627 } // library marker kkossev.commonLib, line 628 } // library marker kkossev.commonLib, line 629 catch (e2) { // library marker kkossev.commonLib, line 630 logWarn "exception ${e2} caught while parsing using an alternative method myParseDescriptionAsMap description: ${description}" // library marker kkossev.commonLib, line 631 return [:] // library marker kkossev.commonLib, line 632 } // library marker kkossev.commonLib, line 633 logDebug "alternative method parsing success: descMap=${descMap}" // library marker kkossev.commonLib, line 634 } // library marker kkossev.commonLib, line 635 return descMap // library marker kkossev.commonLib, line 636 } // library marker kkossev.commonLib, line 637 // return true if the messages is processed here, and further processing in the main parse method should be cancelled ! // library marker kkossev.commonLib, line 639 // return false if the cluster is not a Tuya cluster // library marker kkossev.commonLib, line 640 private boolean isTuyaE00xCluster(String description) { // library marker kkossev.commonLib, line 641 if (description == null || !(description.indexOf('cluster: E000') >= 0 || description.indexOf('cluster: E001') >= 0)) { // library marker kkossev.commonLib, line 642 return false // library marker kkossev.commonLib, line 643 } // library marker kkossev.commonLib, line 644 // try to parse ... // library marker kkossev.commonLib, line 645 //logDebug "Tuya cluster: E000 or E001 - try to parse it..." // library marker kkossev.commonLib, line 646 Map descMap = [:] // library marker kkossev.commonLib, line 647 try { // library marker kkossev.commonLib, line 648 descMap = zigbee.parseDescriptionAsMap(description) // library marker kkossev.commonLib, line 649 logDebug "TuyaE00xCluster Desc Map: ${descMap}" // library marker kkossev.commonLib, line 650 } // library marker kkossev.commonLib, line 651 catch (e) { // library marker kkossev.commonLib, line 652 logDebug "exception caught while parsing description: ${description}" // library marker kkossev.commonLib, line 653 logDebug "TuyaE00xCluster Desc Map: ${descMap}" // library marker kkossev.commonLib, line 654 // cluster E001 is the one that is generating exceptions... // library marker kkossev.commonLib, line 655 return true // library marker kkossev.commonLib, line 656 } // library marker kkossev.commonLib, line 657 if (descMap.cluster == 'E000' && descMap.attrId in ['D001', 'D002', 'D003']) { // library marker kkossev.commonLib, line 659 logDebug "Tuya Specific cluster ${descMap.cluster} attribute ${descMap.attrId} value is ${descMap.value}" // library marker kkossev.commonLib, line 660 } // library marker kkossev.commonLib, line 661 else if (descMap.cluster == 'E001' && descMap.attrId == 'D010') { // library marker kkossev.commonLib, line 662 if (settings?.logEnable) { logInfo "power on behavior is ${powerOnBehaviourOptions[safeToInt(descMap.value).toString()]} (${descMap.value})" } // library marker kkossev.commonLib, line 663 } // library marker kkossev.commonLib, line 664 else if (descMap.cluster == 'E001' && descMap.attrId == 'D030') { // library marker kkossev.commonLib, line 665 if (settings?.logEnable) { logInfo "swith type is ${switchTypeOptions[safeToInt(descMap.value).toString()]} (${descMap.value})" } // library marker kkossev.commonLib, line 666 } // library marker kkossev.commonLib, line 667 else { // library marker kkossev.commonLib, line 668 logDebug "unprocessed TuyaE00xCluster Desc Map: $descMap" // library marker kkossev.commonLib, line 669 return false // library marker kkossev.commonLib, line 670 } // library marker kkossev.commonLib, line 671 return true // processed // library marker kkossev.commonLib, line 672 } // library marker kkossev.commonLib, line 673 // return true if processed here, and further processing in the main parse method should be cancelled ! // library marker kkossev.commonLib, line 675 private boolean otherTuyaOddities(final String description) { // library marker kkossev.commonLib, line 676 /* // library marker kkossev.commonLib, line 677 if (description.indexOf('cluster: 0000') >= 0 && description.indexOf('attrId: 0004') >= 0) { // library marker kkossev.commonLib, line 678 if (logEnable) log.debug "${device.displayName} skipping Tuya parse of cluster 0 attrId 4" // parseDescriptionAsMap throws exception when processing Tuya cluster 0 attrId 4 // library marker kkossev.commonLib, line 679 return true // library marker kkossev.commonLib, line 680 } // library marker kkossev.commonLib, line 681 */ // library marker kkossev.commonLib, line 682 Map descMap = [:] // library marker kkossev.commonLib, line 683 try { // library marker kkossev.commonLib, line 684 descMap = zigbee.parseDescriptionAsMap(description) // library marker kkossev.commonLib, line 685 } // library marker kkossev.commonLib, line 686 catch (e1) { // library marker kkossev.commonLib, line 687 logWarn "exception ${e1} caught while parseDescriptionAsMap otherTuyaOddities description: ${description}" // library marker kkossev.commonLib, line 688 // try alternative custom parsing // library marker kkossev.commonLib, line 689 descMap = [:] // library marker kkossev.commonLib, line 690 try { // library marker kkossev.commonLib, line 691 descMap += description.replaceAll('\\[|\\]', '').split(',').collectEntries { entry -> // library marker kkossev.commonLib, line 692 List pair = entry.split(':') // library marker kkossev.commonLib, line 693 [(pair.first().trim()): pair.last().trim()] // library marker kkossev.commonLib, line 694 } // library marker kkossev.commonLib, line 695 } // library marker kkossev.commonLib, line 696 catch (e2) { // library marker kkossev.commonLib, line 697 logWarn "exception ${e2} caught while parsing using an alternative method otherTuyaOddities description: ${description}" // library marker kkossev.commonLib, line 698 return true // library marker kkossev.commonLib, line 699 } // library marker kkossev.commonLib, line 700 logDebug "alternative method parsing success: descMap=${descMap}" // library marker kkossev.commonLib, line 701 } // library marker kkossev.commonLib, line 702 //if (logEnable) {log.trace "${device.displayName} Checking Tuya Oddities Desc Map: $descMap"} // library marker kkossev.commonLib, line 703 if (descMap.attrId == null) { // library marker kkossev.commonLib, line 704 //logDebug "otherTuyaOddities: descMap = ${descMap}" // library marker kkossev.commonLib, line 705 //if (logEnable) log.trace "${device.displayName} otherTuyaOddities - Cluster ${descMap.clusterId} NO ATTRIBUTE, skipping" // library marker kkossev.commonLib, line 706 return false // library marker kkossev.commonLib, line 707 } // library marker kkossev.commonLib, line 708 boolean bWasAtLeastOneAttributeProcessed = false // library marker kkossev.commonLib, line 709 boolean bWasThereAnyStandardAttribite = false // library marker kkossev.commonLib, line 710 // attribute report received // library marker kkossev.commonLib, line 711 List attrData = [[cluster: descMap.cluster ,attrId: descMap.attrId, value: descMap.value, status: descMap.status]] // library marker kkossev.commonLib, line 712 descMap.additionalAttrs.each { // library marker kkossev.commonLib, line 713 attrData << [cluster: descMap.cluster, attrId: it.attrId, value: it.value, status: it.status] // library marker kkossev.commonLib, line 714 } // library marker kkossev.commonLib, line 715 attrData.each { // library marker kkossev.commonLib, line 716 if (it.status == '86') { // library marker kkossev.commonLib, line 717 logWarn "Tuya Cluster ${descMap.cluster} unsupported attrId ${it.attrId}" // library marker kkossev.commonLib, line 718 // TODO - skip parsing? // library marker kkossev.commonLib, line 719 } // library marker kkossev.commonLib, line 720 switch (it.cluster) { // library marker kkossev.commonLib, line 721 case '0000' : // library marker kkossev.commonLib, line 722 if (it.attrId in ['FFE0', 'FFE1', 'FFE2', 'FFE4']) { // library marker kkossev.commonLib, line 723 logTrace "Cluster ${descMap.cluster} Tuya specific attrId ${it.attrId} value ${it.value})" // library marker kkossev.commonLib, line 724 bWasAtLeastOneAttributeProcessed = true // library marker kkossev.commonLib, line 725 } // library marker kkossev.commonLib, line 726 else if (it.attrId in ['FFFE', 'FFDF']) { // library marker kkossev.commonLib, line 727 logTrace "Cluster ${descMap.cluster} Tuya specific attrId ${it.attrId} value ${it.value})" // library marker kkossev.commonLib, line 728 bWasAtLeastOneAttributeProcessed = true // library marker kkossev.commonLib, line 729 } // library marker kkossev.commonLib, line 730 else { // library marker kkossev.commonLib, line 731 //logDebug "otherTuyaOddities? - Cluster ${descMap.cluster} attrId ${it.attrId} value ${it.value}) N/A, skipping" // library marker kkossev.commonLib, line 732 bWasThereAnyStandardAttribite = true // library marker kkossev.commonLib, line 733 } // library marker kkossev.commonLib, line 734 break // library marker kkossev.commonLib, line 735 default : // library marker kkossev.commonLib, line 736 //if (logEnable) log.trace "${device.displayName} otherTuyaOddities - Cluster ${it.cluster} N/A, skipping" // library marker kkossev.commonLib, line 737 break // library marker kkossev.commonLib, line 738 } // switch // library marker kkossev.commonLib, line 739 } // for each attribute // library marker kkossev.commonLib, line 740 return bWasAtLeastOneAttributeProcessed && !bWasThereAnyStandardAttribite // library marker kkossev.commonLib, line 741 } // library marker kkossev.commonLib, line 742 public String intTo16bitUnsignedHex(int value) { // library marker kkossev.commonLib, line 744 String hexStr = zigbee.convertToHexString(value.toInteger(), 4) // library marker kkossev.commonLib, line 745 return new String(hexStr.substring(2, 4) + hexStr.substring(0, 2)) // library marker kkossev.commonLib, line 746 } // library marker kkossev.commonLib, line 747 public String intTo8bitUnsignedHex(int value) { // library marker kkossev.commonLib, line 749 return zigbee.convertToHexString(value.toInteger(), 2) // library marker kkossev.commonLib, line 750 } // library marker kkossev.commonLib, line 751 /* // library marker kkossev.commonLib, line 753 * ----------------------------------------------------------------------------- // library marker kkossev.commonLib, line 754 * Tuya cluster EF00 specific code // library marker kkossev.commonLib, line 755 * ----------------------------------------------------------------------------- // library marker kkossev.commonLib, line 756 */ // library marker kkossev.commonLib, line 757 private static int getCLUSTER_TUYA() { 0xEF00 } // library marker kkossev.commonLib, line 758 private static int getSETDATA() { 0x00 } // library marker kkossev.commonLib, line 759 private static int getSETTIME() { 0x24 } // library marker kkossev.commonLib, line 760 // Tuya Commands // library marker kkossev.commonLib, line 762 private static int getTUYA_REQUEST() { 0x00 } // library marker kkossev.commonLib, line 763 private static int getTUYA_REPORTING() { 0x01 } // library marker kkossev.commonLib, line 764 private static int getTUYA_QUERY() { 0x02 } // library marker kkossev.commonLib, line 765 private static int getTUYA_STATUS_SEARCH() { 0x06 } // library marker kkossev.commonLib, line 766 private static int getTUYA_TIME_SYNCHRONISATION() { 0x24 } // library marker kkossev.commonLib, line 767 // tuya DP type // library marker kkossev.commonLib, line 769 private static String getDP_TYPE_RAW() { '01' } // [ bytes ] // library marker kkossev.commonLib, line 770 private static String getDP_TYPE_BOOL() { '01' } // [ 0/1 ] // library marker kkossev.commonLib, line 771 private static String getDP_TYPE_VALUE() { '02' } // [ 4 byte value ] // library marker kkossev.commonLib, line 772 private static String getDP_TYPE_STRING() { '03' } // [ N byte string ] // library marker kkossev.commonLib, line 773 private static String getDP_TYPE_ENUM() { '04' } // [ 0-255 ] // library marker kkossev.commonLib, line 774 private static String getDP_TYPE_BITMAP() { '05' } // [ 1,2,4 bytes ] as bits // library marker kkossev.commonLib, line 775 private void syncTuyaDateTime() { // library marker kkossev.commonLib, line 777 // The data format for time synchronization, including standard timestamps and local timestamps. Standard timestamp (4 bytes) local timestamp (4 bytes) Time synchronization data format: The standard timestamp is the total number of seconds from 00:00:00 on January 01, 1970 GMT to the present. // library marker kkossev.commonLib, line 778 // For example, local timestamp = standard timestamp + number of seconds between standard time and local time (including time zone and daylight saving time). // Y2K = 946684800 // library marker kkossev.commonLib, line 779 long offset = 0 // library marker kkossev.commonLib, line 780 int offsetHours = 0 // library marker kkossev.commonLib, line 781 Calendar cal = Calendar.getInstance() //it return same time as new Date() // library marker kkossev.commonLib, line 782 int hour = cal.get(Calendar.HOUR_OF_DAY) // library marker kkossev.commonLib, line 783 try { // library marker kkossev.commonLib, line 784 offset = location.getTimeZone().getOffset(new Date().getTime()) // library marker kkossev.commonLib, line 785 offsetHours = (offset / 3600000) as int // library marker kkossev.commonLib, line 786 logDebug "timezone offset of current location is ${offset} (${offsetHours} hours), current hour is ${hour} h" // library marker kkossev.commonLib, line 787 } catch (e) { // library marker kkossev.commonLib, line 788 log.error "${device.displayName} cannot resolve current location. please set location in Hubitat location setting. Setting timezone offset to zero" // library marker kkossev.commonLib, line 789 } // library marker kkossev.commonLib, line 790 // // library marker kkossev.commonLib, line 791 List cmds = zigbee.command(CLUSTER_TUYA, SETTIME, '0008' + zigbee.convertToHexString((int)(now() / 1000), 8) + zigbee.convertToHexString((int)((now() + offset) / 1000), 8)) // library marker kkossev.commonLib, line 792 sendZigbeeCommands(cmds) // library marker kkossev.commonLib, line 793 logDebug "Tuya device time synchronized to ${unix2formattedDate(now())} (${cmds})" // library marker kkossev.commonLib, line 794 } // library marker kkossev.commonLib, line 795 // called from the main parse method when the cluster is 0xEF00 and no custom handler is defined // library marker kkossev.commonLib, line 797 public void standardParseTuyaCluster(final Map descMap) { // library marker kkossev.commonLib, line 798 if (descMap?.clusterInt == CLUSTER_TUYA && descMap?.command == '24') { //getSETTIME // library marker kkossev.commonLib, line 799 syncTuyaDateTime() // library marker kkossev.commonLib, line 800 } // library marker kkossev.commonLib, line 801 else if (descMap?.clusterInt == CLUSTER_TUYA && descMap?.command == '0B') { // ZCL Command Default Response // library marker kkossev.commonLib, line 802 String clusterCmd = descMap?.data[0] // library marker kkossev.commonLib, line 803 String status = descMap?.data[1] // library marker kkossev.commonLib, line 804 logDebug "device has received Tuya cluster ZCL command 0x${clusterCmd} response 0x${status} data = ${descMap?.data}" // library marker kkossev.commonLib, line 805 if (status != '00') { // library marker kkossev.commonLib, line 806 logWarn "ATTENTION! manufacturer = ${device.getDataValue('manufacturer')} unsupported Tuya cluster ZCL command 0x${clusterCmd} response 0x${status} data = ${descMap?.data} !!!" // library marker kkossev.commonLib, line 807 } // library marker kkossev.commonLib, line 808 } // library marker kkossev.commonLib, line 809 else if ((descMap?.clusterInt == CLUSTER_TUYA) && (descMap?.command == '01' || descMap?.command == '02' || descMap?.command == '05' || descMap?.command == '06')) { // library marker kkossev.commonLib, line 810 int dataLen = descMap?.data.size() // library marker kkossev.commonLib, line 811 //log.warn "dataLen=${dataLen}" // library marker kkossev.commonLib, line 812 //def transid = zigbee.convertHexToInt(descMap?.data[1]) // "transid" is just a "counter", a response will have the same transid as the command // library marker kkossev.commonLib, line 813 if (dataLen <= 5) { // library marker kkossev.commonLib, line 814 logWarn "unprocessed short Tuya command response: dp_id=${descMap?.data[3]} dp=${descMap?.data[2]} fncmd_len=${fncmd_len} data=${descMap?.data})" // library marker kkossev.commonLib, line 815 return // library marker kkossev.commonLib, line 816 } // library marker kkossev.commonLib, line 817 boolean isSpammyDeviceProfileDefined = this.respondsTo('isSpammyDeviceProfile') // check if the method exists 05/21/2024 // library marker kkossev.commonLib, line 818 for (int i = 0; i < (dataLen - 4); ) { // library marker kkossev.commonLib, line 819 int dp = zigbee.convertHexToInt(descMap?.data[2 + i]) // "dp" field describes the action/message of a command frame // library marker kkossev.commonLib, line 820 int dp_id = zigbee.convertHexToInt(descMap?.data[3 + i]) // "dp_identifier" is device dependant // library marker kkossev.commonLib, line 821 int fncmd_len = zigbee.convertHexToInt(descMap?.data[5 + i]) // library marker kkossev.commonLib, line 822 int fncmd = getTuyaAttributeValue(descMap?.data, i) // // library marker kkossev.commonLib, line 823 if (!isChattyDeviceReport(descMap) && isSpammyDeviceProfileDefined && !isSpammyDeviceProfile()) { // library marker kkossev.commonLib, line 824 logDebug "standardParseTuyaCluster: command=${descMap?.command} dp_id=${dp_id} dp=${dp} (0x${descMap?.data[2 + i]}) fncmd=${fncmd} fncmd_len=${fncmd_len} (index=${i})" // library marker kkossev.commonLib, line 825 } // library marker kkossev.commonLib, line 826 standardProcessTuyaDP(descMap, dp, dp_id, fncmd) // library marker kkossev.commonLib, line 827 i = i + fncmd_len + 4 // library marker kkossev.commonLib, line 828 } // library marker kkossev.commonLib, line 829 } // library marker kkossev.commonLib, line 830 else { // library marker kkossev.commonLib, line 831 logWarn "standardParseTuyaCluster: unprocessed Tuya cluster command ${descMap?.command} data=${descMap?.data}" // library marker kkossev.commonLib, line 832 } // library marker kkossev.commonLib, line 833 } // library marker kkossev.commonLib, line 834 // called from the standardParseTuyaCluster method for each DP chunk in the messages (usually one, but could be multiple DPs in one message) // library marker kkossev.commonLib, line 836 void standardProcessTuyaDP(final Map descMap, final int dp, final int dp_id, final int fncmd, final int dp_len=0) { // library marker kkossev.commonLib, line 837 logTrace "standardProcessTuyaDP: checking customProcessTuyaDp dp=${dp} dp_id=${dp_id} fncmd=${fncmd} dp_len=${dp_len}" // library marker kkossev.commonLib, line 838 if (this.respondsTo('customProcessTuyaDp')) { // library marker kkossev.commonLib, line 839 logTrace 'standardProcessTuyaDP: customProcessTuyaDp exists, calling it...' // library marker kkossev.commonLib, line 840 if (customProcessTuyaDp(descMap, dp, dp_id, fncmd, dp_len) == true) { // library marker kkossev.commonLib, line 841 return // EF00 DP has been processed in the custom handler - we are done! // library marker kkossev.commonLib, line 842 } // library marker kkossev.commonLib, line 843 } // library marker kkossev.commonLib, line 844 // check if DeviceProfile processing method exists (deviceProfieLib should be included in the main driver) // library marker kkossev.commonLib, line 845 if (this.respondsTo(processTuyaDPfromDeviceProfile)) { // library marker kkossev.commonLib, line 846 if (processTuyaDPfromDeviceProfile(descMap, dp, dp_id, fncmd, dp_len) == true) { // library marker kkossev.commonLib, line 847 return // sucessfuly processed the new way - we are done. (version 3.0) // library marker kkossev.commonLib, line 848 } // library marker kkossev.commonLib, line 849 } // library marker kkossev.commonLib, line 850 logWarn "NOT PROCESSED Tuya cmd: dp=${dp} value=${fncmd} descMap.data = ${descMap?.data}" // library marker kkossev.commonLib, line 851 } // library marker kkossev.commonLib, line 852 public int getTuyaAttributeValue(final List _data, final int index) { // library marker kkossev.commonLib, line 854 int retValue = 0 // library marker kkossev.commonLib, line 855 if (_data.size() >= 6) { // library marker kkossev.commonLib, line 856 int dataLength = zigbee.convertHexToInt(_data[5 + index]) // library marker kkossev.commonLib, line 857 if (dataLength == 0) { return 0 } // library marker kkossev.commonLib, line 858 int power = 1 // library marker kkossev.commonLib, line 859 for (i in dataLength..1) { // library marker kkossev.commonLib, line 860 retValue = retValue + power * zigbee.convertHexToInt(_data[index + i + 5]) // library marker kkossev.commonLib, line 861 power = power * 256 // library marker kkossev.commonLib, line 862 } // library marker kkossev.commonLib, line 863 } // library marker kkossev.commonLib, line 864 return retValue // library marker kkossev.commonLib, line 865 } // library marker kkossev.commonLib, line 866 public List getTuyaCommand(String dp, String dp_type, String fncmd, int tuyaCmdDefault = SETDATA) { return sendTuyaCommand(dp, dp_type, fncmd, tuyaCmdDefault) } // library marker kkossev.commonLib, line 868 public List sendTuyaCommand(String dp, String dp_type, String fncmd, int tuyaCmdDefault = SETDATA) { // library marker kkossev.commonLib, line 870 List cmds = [] // library marker kkossev.commonLib, line 871 int ep = safeToInt(state.destinationEP) // library marker kkossev.commonLib, line 872 if (ep == null || ep == 0) { ep = 1 } // library marker kkossev.commonLib, line 873 int tuyaCmd // library marker kkossev.commonLib, line 874 // added 07/01/2024 - deviceProfilesV3 device key tuyaCmd:04 : owerwrite all sendTuyaCommand calls for a specfic device profile, if specified! // library marker kkossev.commonLib, line 875 if (this.respondsTo('getDEVICE') && DEVICE?.device?.tuyaCmd != null) { // library marker kkossev.commonLib, line 876 tuyaCmd = DEVICE?.device?.tuyaCmd // library marker kkossev.commonLib, line 877 } // library marker kkossev.commonLib, line 878 else { // library marker kkossev.commonLib, line 879 tuyaCmd = tuyaCmdDefault // 0x00 is the default command for most of the Tuya devices, except some .. // library marker kkossev.commonLib, line 880 } // library marker kkossev.commonLib, line 881 cmds = zigbee.command(CLUSTER_TUYA, tuyaCmd, [destEndpoint :ep], delay = 201, PACKET_ID + dp + dp_type + zigbee.convertToHexString((int)(fncmd.length() / 2), 4) + fncmd ) // library marker kkossev.commonLib, line 882 logDebug "${device.displayName} getTuyaCommand (dp=$dp fncmd=$fncmd dp_type=$dp_type) = ${cmds}" // library marker kkossev.commonLib, line 883 return cmds // library marker kkossev.commonLib, line 884 } // library marker kkossev.commonLib, line 885 private String getPACKET_ID() { return zigbee.convertToHexString(new Random().nextInt(65536), 4) } // library marker kkossev.commonLib, line 887 public void tuyaTest(String dpCommand, String dpValue, String dpTypeString ) { // library marker kkossev.commonLib, line 889 String dpType = dpTypeString == 'DP_TYPE_VALUE' ? DP_TYPE_VALUE : dpTypeString == 'DP_TYPE_BOOL' ? DP_TYPE_BOOL : dpTypeString == 'DP_TYPE_ENUM' ? DP_TYPE_ENUM : null // library marker kkossev.commonLib, line 890 String dpValHex = dpTypeString == 'DP_TYPE_VALUE' ? zigbee.convertToHexString(dpValue as int, 8) : dpValue // library marker kkossev.commonLib, line 891 if (settings?.logEnable) { log.warn "${device.displayName} sending TEST command=${dpCommand} value=${dpValue} ($dpValHex) type=${dpType}" } // library marker kkossev.commonLib, line 892 sendZigbeeCommands( sendTuyaCommand(dpCommand, dpType, dpValHex) ) // library marker kkossev.commonLib, line 893 } // library marker kkossev.commonLib, line 894 public List tuyaBlackMagic() { // library marker kkossev.commonLib, line 897 int ep = safeToInt(state.destinationEP ?: 01) // library marker kkossev.commonLib, line 898 if (ep == null || ep == 0) { ep = 1 } // library marker kkossev.commonLib, line 899 logInfo 'tuyaBlackMagic()...' // library marker kkossev.commonLib, line 900 return zigbee.readAttribute(0x0000, [0x0004, 0x000, 0x0001, 0x0005, 0x0007, 0xfffe], [destEndpoint :ep], delay = 200) // library marker kkossev.commonLib, line 901 } // library marker kkossev.commonLib, line 902 public List queryAllTuyaDP() { // library marker kkossev.commonLib, line 904 logTrace 'queryAllTuyaDP()' // library marker kkossev.commonLib, line 905 List cmds = zigbee.command(0xEF00, 0x03) // library marker kkossev.commonLib, line 906 return cmds // library marker kkossev.commonLib, line 907 } // library marker kkossev.commonLib, line 908 public void aqaraBlackMagic() { // library marker kkossev.commonLib, line 910 List cmds = [] // library marker kkossev.commonLib, line 911 if (this.respondsTo('customAqaraBlackMagic')) { // library marker kkossev.commonLib, line 912 cmds = customAqaraBlackMagic() // library marker kkossev.commonLib, line 913 } // library marker kkossev.commonLib, line 914 if (cmds != null && !cmds.isEmpty()) { // library marker kkossev.commonLib, line 915 logDebug 'sending aqaraBlackMagic()' // library marker kkossev.commonLib, line 916 sendZigbeeCommands(cmds) // library marker kkossev.commonLib, line 917 return // library marker kkossev.commonLib, line 918 } // library marker kkossev.commonLib, line 919 logDebug 'aqaraBlackMagic() was SKIPPED' // library marker kkossev.commonLib, line 920 } // library marker kkossev.commonLib, line 921 // Invoked from configure() // library marker kkossev.commonLib, line 923 public List initializeDevice() { // library marker kkossev.commonLib, line 924 List cmds = [] // library marker kkossev.commonLib, line 925 logInfo 'initializeDevice...' // library marker kkossev.commonLib, line 926 if (this.respondsTo('customInitializeDevice')) { // library marker kkossev.commonLib, line 927 List customCmds = customInitializeDevice() // library marker kkossev.commonLib, line 928 if (customCmds != null && !customCmds.isEmpty()) { cmds += customCmds } // library marker kkossev.commonLib, line 929 } // library marker kkossev.commonLib, line 930 else { logDebug 'no customInitializeDevice method defined' } // library marker kkossev.commonLib, line 931 logDebug "initializeDevice(): cmds=${cmds}" // library marker kkossev.commonLib, line 932 return cmds // library marker kkossev.commonLib, line 933 } // library marker kkossev.commonLib, line 934 // Invoked from configure() // library marker kkossev.commonLib, line 936 public List configureDevice() { // library marker kkossev.commonLib, line 937 List cmds = [] // library marker kkossev.commonLib, line 938 logInfo 'configureDevice...' // library marker kkossev.commonLib, line 939 if (this.respondsTo('customConfigureDevice')) { // library marker kkossev.commonLib, line 940 List customCmds = customConfigureDevice() // library marker kkossev.commonLib, line 941 if (customCmds != null && !customCmds.isEmpty()) { cmds += customCmds } // library marker kkossev.commonLib, line 942 } // library marker kkossev.commonLib, line 943 else { logDebug 'no customConfigureDevice method defined' } // library marker kkossev.commonLib, line 944 // sendZigbeeCommands(cmds) changed 03/04/2024 // library marker kkossev.commonLib, line 945 logDebug "configureDevice(): cmds=${cmds}" // library marker kkossev.commonLib, line 946 return cmds // library marker kkossev.commonLib, line 947 } // library marker kkossev.commonLib, line 948 /* // library marker kkossev.commonLib, line 950 * ----------------------------------------------------------------------------- // library marker kkossev.commonLib, line 951 * Hubitat default handlers methods // library marker kkossev.commonLib, line 952 * ----------------------------------------------------------------------------- // library marker kkossev.commonLib, line 953 */ // library marker kkossev.commonLib, line 954 List customHandlers(final List customHandlersList) { // library marker kkossev.commonLib, line 956 List cmds = [] // library marker kkossev.commonLib, line 957 if (customHandlersList != null && !customHandlersList.isEmpty()) { // library marker kkossev.commonLib, line 958 customHandlersList.each { handler -> // library marker kkossev.commonLib, line 959 if (this.respondsTo(handler)) { // library marker kkossev.commonLib, line 960 List customCmds = this."${handler}"() // library marker kkossev.commonLib, line 961 if (customCmds != null && !customCmds.isEmpty()) { cmds += customCmds } // library marker kkossev.commonLib, line 962 } // library marker kkossev.commonLib, line 963 } // library marker kkossev.commonLib, line 964 } // library marker kkossev.commonLib, line 965 return cmds // library marker kkossev.commonLib, line 966 } // library marker kkossev.commonLib, line 967 public void refresh() { // library marker kkossev.commonLib, line 969 logDebug "refresh()... DEVICE_TYPE is ${DEVICE_TYPE} model=${device.getDataValue('model')} manufacturer=${device.getDataValue('manufacturer')}" // library marker kkossev.commonLib, line 970 checkDriverVersion(state) // library marker kkossev.commonLib, line 971 List cmds = [], customCmds = [] // library marker kkossev.commonLib, line 972 if (this.respondsTo('customRefresh')) { // if there is a customRefresh() method defined in the main driver, call it // library marker kkossev.commonLib, line 973 customCmds = customRefresh() // library marker kkossev.commonLib, line 974 if (customCmds != null && !customCmds.isEmpty()) { cmds += customCmds } else { logDebug 'no customRefresh method defined' } // library marker kkossev.commonLib, line 975 } // library marker kkossev.commonLib, line 976 else { // call all known libraryRefresh methods // library marker kkossev.commonLib, line 977 customCmds = customHandlers(['onOffRefresh', 'groupsRefresh', 'batteryRefresh', 'levelRefresh', 'temperatureRefresh', 'humidityRefresh', 'illuminanceRefresh']) // library marker kkossev.commonLib, line 978 if (customCmds != null && !customCmds.isEmpty()) { cmds += customCmds } else { logDebug 'no libraries refresh() defined' } // library marker kkossev.commonLib, line 979 } // library marker kkossev.commonLib, line 980 if (cmds != null && !cmds.isEmpty()) { // library marker kkossev.commonLib, line 981 logDebug "refresh() cmds=${cmds}" // library marker kkossev.commonLib, line 982 setRefreshRequest() // 3 seconds // library marker kkossev.commonLib, line 983 sendZigbeeCommands(cmds) // library marker kkossev.commonLib, line 984 } // library marker kkossev.commonLib, line 985 else { // library marker kkossev.commonLib, line 986 logDebug "no refresh() commands defined for device type ${DEVICE_TYPE}" // library marker kkossev.commonLib, line 987 } // library marker kkossev.commonLib, line 988 } // library marker kkossev.commonLib, line 989 public void setRefreshRequest() { if (state.states == null) { state.states = [:] } ; state.states['isRefresh'] = true; runInMillis(REFRESH_TIMER, 'clearRefreshRequest', [overwrite: true]) } // library marker kkossev.commonLib, line 991 public void clearRefreshRequest() { if (state.states == null) { state.states = [:] } ; state.states['isRefresh'] = false } // library marker kkossev.commonLib, line 992 public void clearInfoEvent() { sendInfoEvent('clear') } // library marker kkossev.commonLib, line 993 public void sendInfoEvent(String info=null) { // library marker kkossev.commonLib, line 995 if (info == null || info == 'clear') { // library marker kkossev.commonLib, line 996 logDebug 'clearing the Status event' // library marker kkossev.commonLib, line 997 sendEvent(name: 'Status', value: 'clear', type: 'digital') // library marker kkossev.commonLib, line 998 } // library marker kkossev.commonLib, line 999 else { // library marker kkossev.commonLib, line 1000 logInfo "${info}" // library marker kkossev.commonLib, line 1001 sendEvent(name: 'Status', value: info, type: 'digital') // library marker kkossev.commonLib, line 1002 runIn(INFO_AUTO_CLEAR_PERIOD, 'clearInfoEvent') // automatically clear the Info attribute after 1 minute // library marker kkossev.commonLib, line 1003 } // library marker kkossev.commonLib, line 1004 } // library marker kkossev.commonLib, line 1005 public void ping() { // library marker kkossev.commonLib, line 1007 if (state.lastTx == null ) { state.lastTx = [:] } ; state.lastTx['pingTime'] = new Date().getTime() // library marker kkossev.commonLib, line 1008 if (state.states == null ) { state.states = [:] } ; state.states['isPing'] = true // library marker kkossev.commonLib, line 1009 scheduleCommandTimeoutCheck() // library marker kkossev.commonLib, line 1010 int pingAttr = (device.getDataValue('manufacturer') == 'SONOFF') ? 0x05 : PING_ATTR_ID // library marker kkossev.commonLib, line 1011 if (isVirtual()) { runInMillis(10, 'virtualPong') } // library marker kkossev.commonLib, line 1012 else { sendZigbeeCommands(zigbee.readAttribute(zigbee.BASIC_CLUSTER, pingAttr, [:], 0) ) } // library marker kkossev.commonLib, line 1013 logDebug 'ping...' // library marker kkossev.commonLib, line 1014 } // library marker kkossev.commonLib, line 1015 private void virtualPong() { // library marker kkossev.commonLib, line 1017 logDebug 'virtualPing: pong!' // library marker kkossev.commonLib, line 1018 Long now = new Date().getTime() // library marker kkossev.commonLib, line 1019 int timeRunning = now.toInteger() - (state.lastTx['pingTime'] ?: '0').toInteger() // library marker kkossev.commonLib, line 1020 if (timeRunning > 0 && timeRunning < MAX_PING_MILISECONDS) { // library marker kkossev.commonLib, line 1021 state.stats['pingsOK'] = (state.stats['pingsOK'] ?: 0) + 1 // library marker kkossev.commonLib, line 1022 if (timeRunning < safeToInt((state.stats['pingsMin'] ?: '9999'))) { state.stats['pingsMin'] = timeRunning } // library marker kkossev.commonLib, line 1023 if (timeRunning > safeToInt((state.stats['pingsMax'] ?: '0'))) { state.stats['pingsMax'] = timeRunning } // library marker kkossev.commonLib, line 1024 state.stats['pingsAvg'] = approxRollingAverage(safeToDouble(state.stats['pingsAvg']), safeToDouble(timeRunning)) as int // library marker kkossev.commonLib, line 1025 sendRttEvent() // library marker kkossev.commonLib, line 1026 } // library marker kkossev.commonLib, line 1027 else { // library marker kkossev.commonLib, line 1028 logWarn "unexpected ping timeRunning=${timeRunning} " // library marker kkossev.commonLib, line 1029 } // library marker kkossev.commonLib, line 1030 state.states['isPing'] = false // library marker kkossev.commonLib, line 1031 unscheduleCommandTimeoutCheck(state) // library marker kkossev.commonLib, line 1032 } // library marker kkossev.commonLib, line 1033 public void sendRttEvent( String value=null) { // library marker kkossev.commonLib, line 1035 Long now = new Date().getTime() // library marker kkossev.commonLib, line 1036 if (state.lastTx == null ) { state.lastTx = [:] } // library marker kkossev.commonLib, line 1037 int timeRunning = now.toInteger() - (state.lastTx['pingTime'] ?: now).toInteger() // library marker kkossev.commonLib, line 1038 String descriptionText = "Round-trip time is ${timeRunning} ms (min=${state.stats['pingsMin']} max=${state.stats['pingsMax']} average=${state.stats['pingsAvg']})" // library marker kkossev.commonLib, line 1039 if (value == null) { // library marker kkossev.commonLib, line 1040 logInfo "${descriptionText}" // library marker kkossev.commonLib, line 1041 sendEvent(name: 'rtt', value: timeRunning, descriptionText: descriptionText, unit: 'ms', type: 'physical') // library marker kkossev.commonLib, line 1042 } // library marker kkossev.commonLib, line 1043 else { // library marker kkossev.commonLib, line 1044 descriptionText = "Round-trip time : ${value}" // library marker kkossev.commonLib, line 1045 logInfo "${descriptionText}" // library marker kkossev.commonLib, line 1046 sendEvent(name: 'rtt', value: value, descriptionText: descriptionText, type: 'physical') // library marker kkossev.commonLib, line 1047 } // library marker kkossev.commonLib, line 1048 } // library marker kkossev.commonLib, line 1049 private String clusterLookup(final Object cluster) { // library marker kkossev.commonLib, line 1051 if (cluster != null) { // library marker kkossev.commonLib, line 1052 return zigbee.clusterLookup(cluster.toInteger()) ?: "private cluster 0x${intToHexStr(cluster.toInteger())}" // library marker kkossev.commonLib, line 1053 } // library marker kkossev.commonLib, line 1054 logWarn 'cluster is NULL!' // library marker kkossev.commonLib, line 1055 return 'NULL' // library marker kkossev.commonLib, line 1056 } // library marker kkossev.commonLib, line 1057 private void scheduleCommandTimeoutCheck(int delay = COMMAND_TIMEOUT) { // library marker kkossev.commonLib, line 1059 if (state.states == null) { state.states = [:] } // library marker kkossev.commonLib, line 1060 state.states['isTimeoutCheck'] = true // library marker kkossev.commonLib, line 1061 runIn(delay, 'deviceCommandTimeout') // library marker kkossev.commonLib, line 1062 } // library marker kkossev.commonLib, line 1063 // unschedule() is a very time consuming operation : ~ 5 milliseconds per call ! // library marker kkossev.commonLib, line 1065 void unscheduleCommandTimeoutCheck(final Map state) { // can not be static :( // library marker kkossev.commonLib, line 1066 if (state.states == null) { state.states = [:] } // library marker kkossev.commonLib, line 1067 if (state.states['isTimeoutCheck'] == true) { // library marker kkossev.commonLib, line 1068 state.states['isTimeoutCheck'] = false // library marker kkossev.commonLib, line 1069 unschedule('deviceCommandTimeout') // library marker kkossev.commonLib, line 1070 } // library marker kkossev.commonLib, line 1071 } // library marker kkossev.commonLib, line 1072 void deviceCommandTimeout() { // library marker kkossev.commonLib, line 1074 logWarn 'no response received (sleepy device or offline?)' // library marker kkossev.commonLib, line 1075 sendRttEvent('timeout') // library marker kkossev.commonLib, line 1076 state.stats['pingsFail'] = (state.stats['pingsFail'] ?: 0) + 1 // library marker kkossev.commonLib, line 1077 if (state.health?.isHealthCheck == true) { // library marker kkossev.commonLib, line 1078 logWarn 'device health check failed!' // library marker kkossev.commonLib, line 1079 state.health?.checkCtr3 = (state.health?.checkCtr3 ?: 0 ) + 1 // library marker kkossev.commonLib, line 1080 if (state.health?.checkCtr3 >= PRESENCE_COUNT_THRESHOLD) { // library marker kkossev.commonLib, line 1081 if ((device.currentValue('healthStatus') ?: 'unknown') != 'offline' ) { // library marker kkossev.commonLib, line 1082 sendHealthStatusEvent('offline') // library marker kkossev.commonLib, line 1083 } // library marker kkossev.commonLib, line 1084 } // library marker kkossev.commonLib, line 1085 state.health['isHealthCheck'] = false // library marker kkossev.commonLib, line 1086 } // library marker kkossev.commonLib, line 1087 } // library marker kkossev.commonLib, line 1088 private void scheduleDeviceHealthCheck(final int intervalMins, final int healthMethod) { // library marker kkossev.commonLib, line 1090 if (healthMethod == 1 || healthMethod == 2) { // library marker kkossev.commonLib, line 1091 String cron = getCron( intervalMins * 60 ) // library marker kkossev.commonLib, line 1092 schedule(cron, 'deviceHealthCheck') // library marker kkossev.commonLib, line 1093 logDebug "deviceHealthCheck is scheduled every ${intervalMins} minutes" // library marker kkossev.commonLib, line 1094 } // library marker kkossev.commonLib, line 1095 else { // library marker kkossev.commonLib, line 1096 logWarn 'deviceHealthCheck is not scheduled!' // library marker kkossev.commonLib, line 1097 unschedule('deviceHealthCheck') // library marker kkossev.commonLib, line 1098 } // library marker kkossev.commonLib, line 1099 } // library marker kkossev.commonLib, line 1100 private void unScheduleDeviceHealthCheck() { // library marker kkossev.commonLib, line 1102 unschedule('deviceHealthCheck') // library marker kkossev.commonLib, line 1103 device.deleteCurrentState('healthStatus') // library marker kkossev.commonLib, line 1104 logWarn 'device health check is disabled!' // library marker kkossev.commonLib, line 1105 } // library marker kkossev.commonLib, line 1106 // called when any event was received from the Zigbee device in the parse() method. // library marker kkossev.commonLib, line 1108 private void setHealthStatusOnline(Map state) { // library marker kkossev.commonLib, line 1109 if (state.health == null) { state.health = [:] } // library marker kkossev.commonLib, line 1110 state.health['checkCtr3'] = 0 // library marker kkossev.commonLib, line 1111 if (!((device.currentValue('healthStatus') ?: 'unknown') in ['online'])) { // library marker kkossev.commonLib, line 1112 sendHealthStatusEvent('online') // library marker kkossev.commonLib, line 1113 logInfo 'is now online!' // library marker kkossev.commonLib, line 1114 } // library marker kkossev.commonLib, line 1115 } // library marker kkossev.commonLib, line 1116 private void deviceHealthCheck() { // library marker kkossev.commonLib, line 1118 checkDriverVersion(state) // library marker kkossev.commonLib, line 1119 if (state.health == null) { state.health = [:] } // library marker kkossev.commonLib, line 1120 int ctr = state.health['checkCtr3'] ?: 0 // library marker kkossev.commonLib, line 1121 if (ctr >= PRESENCE_COUNT_THRESHOLD) { // library marker kkossev.commonLib, line 1122 if ((device.currentValue('healthStatus') ?: 'unknown') != 'offline' ) { // library marker kkossev.commonLib, line 1123 logWarn 'not present!' // library marker kkossev.commonLib, line 1124 sendHealthStatusEvent('offline') // library marker kkossev.commonLib, line 1125 } // library marker kkossev.commonLib, line 1126 } // library marker kkossev.commonLib, line 1127 else { // library marker kkossev.commonLib, line 1128 logDebug "deviceHealthCheck - online (notPresentCounter=${(ctr + 1)})" // library marker kkossev.commonLib, line 1129 } // library marker kkossev.commonLib, line 1130 state.health['checkCtr3'] = ctr + 1 // library marker kkossev.commonLib, line 1131 // added 03/06/2025 // library marker kkossev.commonLib, line 1132 if (settings?.healthCheckMethod as int == 2) { // library marker kkossev.commonLib, line 1133 state.health['isHealthCheck'] = true // library marker kkossev.commonLib, line 1134 ping() // proactively ping the device... // library marker kkossev.commonLib, line 1135 } // library marker kkossev.commonLib, line 1136 } // library marker kkossev.commonLib, line 1137 private void sendHealthStatusEvent(final String value) { // library marker kkossev.commonLib, line 1139 String descriptionText = "healthStatus changed to ${value}" // library marker kkossev.commonLib, line 1140 sendEvent(name: 'healthStatus', value: value, descriptionText: descriptionText, isStateChange: true, type: 'digital') // library marker kkossev.commonLib, line 1141 if (value == 'online') { // library marker kkossev.commonLib, line 1142 logInfo "${descriptionText}" // library marker kkossev.commonLib, line 1143 } // library marker kkossev.commonLib, line 1144 else { // library marker kkossev.commonLib, line 1145 if (settings?.txtEnable) { log.warn "${device.displayName}} ${descriptionText}" } // library marker kkossev.commonLib, line 1146 } // library marker kkossev.commonLib, line 1147 } // library marker kkossev.commonLib, line 1148 // Invoked by Hubitat when the driver configuration is updated // library marker kkossev.commonLib, line 1150 void updated() { // library marker kkossev.commonLib, line 1151 logInfo 'updated()...' // library marker kkossev.commonLib, line 1152 checkDriverVersion(state) // library marker kkossev.commonLib, line 1153 logInfo"driver version ${driverVersionAndTimeStamp()}" // library marker kkossev.commonLib, line 1154 unschedule() // library marker kkossev.commonLib, line 1155 if (settings.logEnable) { // library marker kkossev.commonLib, line 1157 logTrace(settings.toString()) // library marker kkossev.commonLib, line 1158 runIn(86400, 'logsOff') // library marker kkossev.commonLib, line 1159 } // library marker kkossev.commonLib, line 1160 if (settings.traceEnable) { // library marker kkossev.commonLib, line 1161 logTrace(settings.toString()) // library marker kkossev.commonLib, line 1162 runIn(1800, 'traceOff') // library marker kkossev.commonLib, line 1163 } // library marker kkossev.commonLib, line 1164 final int healthMethod = (settings.healthCheckMethod as Integer) ?: 0 // library marker kkossev.commonLib, line 1166 if (healthMethod == 1 || healthMethod == 2) { // [0: 'Disabled', 1: 'Activity check', 2: 'Periodic polling'] // library marker kkossev.commonLib, line 1167 // schedule the periodic timer // library marker kkossev.commonLib, line 1168 final int interval = (settings.healthCheckInterval as Integer) ?: 0 // library marker kkossev.commonLib, line 1169 if (interval > 0) { // library marker kkossev.commonLib, line 1170 //log.trace "healthMethod=${healthMethod} interval=${interval}" // library marker kkossev.commonLib, line 1171 log.info "scheduling health check every ${interval} minutes by ${HealthcheckMethodOpts.options[healthCheckMethod as int]} method" // library marker kkossev.commonLib, line 1172 scheduleDeviceHealthCheck(interval, healthMethod) // library marker kkossev.commonLib, line 1173 } // library marker kkossev.commonLib, line 1174 } // library marker kkossev.commonLib, line 1175 else { // library marker kkossev.commonLib, line 1176 unScheduleDeviceHealthCheck() // unschedule the periodic job, depending on the healthMethod // library marker kkossev.commonLib, line 1177 log.info 'Health Check is disabled!' // library marker kkossev.commonLib, line 1178 } // library marker kkossev.commonLib, line 1179 if (this.respondsTo('customUpdated')) { // library marker kkossev.commonLib, line 1180 customUpdated() // library marker kkossev.commonLib, line 1181 } // library marker kkossev.commonLib, line 1182 sendInfoEvent('updated') // library marker kkossev.commonLib, line 1184 } // library marker kkossev.commonLib, line 1185 private void logsOff() { // library marker kkossev.commonLib, line 1187 logInfo 'debug logging disabled...' // library marker kkossev.commonLib, line 1188 device.updateSetting('logEnable', [value: 'false', type: 'bool']) // library marker kkossev.commonLib, line 1189 } // library marker kkossev.commonLib, line 1190 private void traceOff() { // library marker kkossev.commonLib, line 1191 logInfo 'trace logging disabled...' // library marker kkossev.commonLib, line 1192 device.updateSetting('traceEnable', [value: 'false', type: 'bool']) // library marker kkossev.commonLib, line 1193 } // library marker kkossev.commonLib, line 1194 public void configure(String command) { // library marker kkossev.commonLib, line 1196 logInfo "configure(${command})..." // library marker kkossev.commonLib, line 1197 if (!(command in (ConfigureOpts.keySet() as List))) { // library marker kkossev.commonLib, line 1198 logWarn "configure: command ${command} must be one of these : ${ConfigureOpts.keySet() as List}" // library marker kkossev.commonLib, line 1199 return // library marker kkossev.commonLib, line 1200 } // library marker kkossev.commonLib, line 1201 // // library marker kkossev.commonLib, line 1202 String func // library marker kkossev.commonLib, line 1203 try { // library marker kkossev.commonLib, line 1204 func = ConfigureOpts[command]?.function // library marker kkossev.commonLib, line 1205 "$func"() // library marker kkossev.commonLib, line 1206 } // library marker kkossev.commonLib, line 1207 catch (e) { // library marker kkossev.commonLib, line 1208 logWarn "Exception ${e} caught while processing $func($value)" // library marker kkossev.commonLib, line 1209 return // library marker kkossev.commonLib, line 1210 } // library marker kkossev.commonLib, line 1211 logInfo "executed '${func}'" // library marker kkossev.commonLib, line 1212 } // library marker kkossev.commonLib, line 1213 /* groovylint-disable-next-line UnusedMethodParameter */ // library marker kkossev.commonLib, line 1215 void configureHelp(final String val) { // library marker kkossev.commonLib, line 1216 if (settings?.txtEnable) { log.warn "${device.displayName} configureHelp: select one of the commands in this list!" } // library marker kkossev.commonLib, line 1217 } // library marker kkossev.commonLib, line 1218 public void loadAllDefaults() { // library marker kkossev.commonLib, line 1220 logDebug 'loadAllDefaults() !!!' // library marker kkossev.commonLib, line 1221 deleteAllSettings() // library marker kkossev.commonLib, line 1222 deleteAllCurrentStates() // library marker kkossev.commonLib, line 1223 deleteAllScheduledJobs() // library marker kkossev.commonLib, line 1224 deleteAllStates() // library marker kkossev.commonLib, line 1225 deleteAllChildDevices() // library marker kkossev.commonLib, line 1226 initialize() // library marker kkossev.commonLib, line 1228 configureNow() // calls also configureDevice() // bug fixed 04/03/2024 // library marker kkossev.commonLib, line 1229 updated() // library marker kkossev.commonLib, line 1230 sendInfoEvent('All Defaults Loaded! F5 to refresh') // library marker kkossev.commonLib, line 1231 } // library marker kkossev.commonLib, line 1232 private void configureNow() { // library marker kkossev.commonLib, line 1234 configure() // library marker kkossev.commonLib, line 1235 } // library marker kkossev.commonLib, line 1236 /** // library marker kkossev.commonLib, line 1238 * Send configuration parameters to the device // library marker kkossev.commonLib, line 1239 * Invoked when device is first installed and when the user updates the configuration TODO // library marker kkossev.commonLib, line 1240 * @return sends zigbee commands // library marker kkossev.commonLib, line 1241 */ // library marker kkossev.commonLib, line 1242 void configure() { // library marker kkossev.commonLib, line 1243 List cmds = [] // library marker kkossev.commonLib, line 1244 if (state.stats == null) { state.stats = [:] } ; state.stats.cfgCtr = (state.stats.cfgCtr ?: 0) + 1 // library marker kkossev.commonLib, line 1245 logInfo "configure()... cfgCtr=${state.stats.cfgCtr}" // library marker kkossev.commonLib, line 1246 logDebug "configure(): settings: $settings" // library marker kkossev.commonLib, line 1247 if (isTuya()) { // library marker kkossev.commonLib, line 1248 cmds += tuyaBlackMagic() // library marker kkossev.commonLib, line 1249 } // library marker kkossev.commonLib, line 1250 aqaraBlackMagic() // zigbee commands are sent here! // library marker kkossev.commonLib, line 1251 List initCmds = initializeDevice() // library marker kkossev.commonLib, line 1252 if (initCmds != null && !initCmds.isEmpty()) { cmds += initCmds } // library marker kkossev.commonLib, line 1253 List cfgCmds = configureDevice() // library marker kkossev.commonLib, line 1254 if (cfgCmds != null && !cfgCmds.isEmpty()) { cmds += cfgCmds } // library marker kkossev.commonLib, line 1255 if (cmds != null && !cmds.isEmpty()) { // library marker kkossev.commonLib, line 1256 sendZigbeeCommands(cmds) // library marker kkossev.commonLib, line 1257 logDebug "configure(): sent cmds = ${cmds}" // library marker kkossev.commonLib, line 1258 sendInfoEvent('sent device configuration') // library marker kkossev.commonLib, line 1259 } // library marker kkossev.commonLib, line 1260 else { // library marker kkossev.commonLib, line 1261 logDebug "configure(): no commands defined for device type ${DEVICE_TYPE}" // library marker kkossev.commonLib, line 1262 } // library marker kkossev.commonLib, line 1263 } // library marker kkossev.commonLib, line 1264 // Invoked when the device is installed with this driver automatically selected. // library marker kkossev.commonLib, line 1266 void installed() { // library marker kkossev.commonLib, line 1267 if (state.stats == null) { state.stats = [:] } ; state.stats.instCtr = (state.stats.instCtr ?: 0) + 1 // library marker kkossev.commonLib, line 1268 logInfo "installed()... instCtr=${state.stats.instCtr}" // library marker kkossev.commonLib, line 1269 // populate some default values for attributes // library marker kkossev.commonLib, line 1270 sendEvent(name: 'healthStatus', value: 'unknown', descriptionText: 'device was installed', type: 'digital') // library marker kkossev.commonLib, line 1271 sendEvent(name: 'powerSource', value: 'unknown', descriptionText: 'device was installed', type: 'digital') // library marker kkossev.commonLib, line 1272 sendInfoEvent('installed') // library marker kkossev.commonLib, line 1273 runIn(3, 'updated') // library marker kkossev.commonLib, line 1274 runIn(5, 'queryPowerSource') // library marker kkossev.commonLib, line 1275 } // library marker kkossev.commonLib, line 1276 private void queryPowerSource() { // library marker kkossev.commonLib, line 1278 sendZigbeeCommands(zigbee.readAttribute(zigbee.BASIC_CLUSTER, 0x0007, [:], 0)) // library marker kkossev.commonLib, line 1279 } // library marker kkossev.commonLib, line 1280 // Invoked from 'LoadAllDefaults' // library marker kkossev.commonLib, line 1282 private void initialize() { // library marker kkossev.commonLib, line 1283 if (state.stats == null) { state.stats = [:] } ; state.stats.initCtr = (state.stats.initCtr ?: 0) + 1 // library marker kkossev.commonLib, line 1284 logInfo "initialize()... initCtr=${state.stats.initCtr}" // library marker kkossev.commonLib, line 1285 if (device.getDataValue('powerSource') == null) { // library marker kkossev.commonLib, line 1286 logInfo "initializing device powerSource 'unknown'" // library marker kkossev.commonLib, line 1287 sendEvent(name: 'powerSource', value: 'unknown', type: 'digital') // library marker kkossev.commonLib, line 1288 } // library marker kkossev.commonLib, line 1289 initializeVars(fullInit = true) // library marker kkossev.commonLib, line 1290 updateTuyaVersion() // library marker kkossev.commonLib, line 1291 updateAqaraVersion() // library marker kkossev.commonLib, line 1292 } // library marker kkossev.commonLib, line 1293 /* // library marker kkossev.commonLib, line 1295 *----------------------------------------------------------------------------- // library marker kkossev.commonLib, line 1296 * kkossev drivers commonly used functions // library marker kkossev.commonLib, line 1297 *----------------------------------------------------------------------------- // library marker kkossev.commonLib, line 1298 */ // library marker kkossev.commonLib, line 1299 static Integer safeToInt(Object val, Integer defaultVal=0) { // library marker kkossev.commonLib, line 1301 return "${val}"?.isInteger() ? "${val}".toInteger() : defaultVal // library marker kkossev.commonLib, line 1302 } // library marker kkossev.commonLib, line 1303 static Double safeToDouble(Object val, Double defaultVal=0.0) { // library marker kkossev.commonLib, line 1305 return "${val}"?.isDouble() ? "${val}".toDouble() : defaultVal // library marker kkossev.commonLib, line 1306 } // library marker kkossev.commonLib, line 1307 static BigDecimal safeToBigDecimal(Object val, BigDecimal defaultVal=0.0) { // library marker kkossev.commonLib, line 1309 return "${val}"?.isBigDecimal() ? "${val}".toBigDecimal() : defaultVal // library marker kkossev.commonLib, line 1310 } // library marker kkossev.commonLib, line 1311 public void sendZigbeeCommands(List cmd) { // library marker kkossev.commonLib, line 1313 if (cmd == null || cmd.isEmpty()) { // library marker kkossev.commonLib, line 1314 logWarn "sendZigbeeCommands: list is empty! cmd=${cmd}" // library marker kkossev.commonLib, line 1315 return // library marker kkossev.commonLib, line 1316 } // library marker kkossev.commonLib, line 1317 hubitat.device.HubMultiAction allActions = new hubitat.device.HubMultiAction() // library marker kkossev.commonLib, line 1318 cmd.each { // library marker kkossev.commonLib, line 1319 if (it == null || it.isEmpty() || it == 'null') { // library marker kkossev.commonLib, line 1320 logWarn "sendZigbeeCommands it: no commands to send! it=${it} (cmd=${cmd})" // library marker kkossev.commonLib, line 1321 return // library marker kkossev.commonLib, line 1322 } // library marker kkossev.commonLib, line 1323 allActions.add(new hubitat.device.HubAction(it, hubitat.device.Protocol.ZIGBEE)) // library marker kkossev.commonLib, line 1324 if (state.stats != null) { state.stats['txCtr'] = (state.stats['txCtr'] ?: 0) + 1 } else { state.stats = [:] } // library marker kkossev.commonLib, line 1325 } // library marker kkossev.commonLib, line 1326 if (state.lastTx != null) { state.lastTx['cmdTime'] = now() } else { state.lastTx = [:] } // library marker kkossev.commonLib, line 1327 sendHubCommand(allActions) // library marker kkossev.commonLib, line 1328 logDebug "sendZigbeeCommands: sent cmd=${cmd}" // library marker kkossev.commonLib, line 1329 } // library marker kkossev.commonLib, line 1330 private String driverVersionAndTimeStamp() { version() + ' ' + timeStamp() + ((_DEBUG) ? ' (debug version!) ' : ' ') + "(${device.getDataValue('model')} ${device.getDataValue('manufacturer')}) (${getModel()} ${location.hub.firmwareVersionString})" } // library marker kkossev.commonLib, line 1332 private String getDeviceInfo() { // library marker kkossev.commonLib, line 1334 return "model=${device.getDataValue('model')} manufacturer=${device.getDataValue('manufacturer')} destinationEP=${state.destinationEP ?: UNKNOWN} deviceProfile=${state.deviceProfile ?: UNKNOWN}" // library marker kkossev.commonLib, line 1335 } // library marker kkossev.commonLib, line 1336 public String getDestinationEP() { // [destEndpoint:safeToInt(getDestinationEP())] // library marker kkossev.commonLib, line 1338 return state.destinationEP ?: device.endpointId ?: '01' // library marker kkossev.commonLib, line 1339 } // library marker kkossev.commonLib, line 1340 //@CompileStatic // library marker kkossev.commonLib, line 1342 public void checkDriverVersion(final Map stateCopy) { // library marker kkossev.commonLib, line 1343 if (stateCopy.driverVersion == null || driverVersionAndTimeStamp() != stateCopy.driverVersion) { // library marker kkossev.commonLib, line 1344 logDebug "checkDriverVersion: updating the settings from the current driver version ${stateCopy.driverVersion} to the new version ${driverVersionAndTimeStamp()}" // library marker kkossev.commonLib, line 1345 sendInfoEvent("Updated to version ${driverVersionAndTimeStamp()}") // library marker kkossev.commonLib, line 1346 state.driverVersion = driverVersionAndTimeStamp() // library marker kkossev.commonLib, line 1347 initializeVars(false) // library marker kkossev.commonLib, line 1348 updateTuyaVersion() // library marker kkossev.commonLib, line 1349 updateAqaraVersion() // library marker kkossev.commonLib, line 1350 } // library marker kkossev.commonLib, line 1351 if (state.states == null) { state.states = [:] } ; if (state.lastRx == null) { state.lastRx = [:] } ; if (state.lastTx == null) { state.lastTx = [:] } ; if (state.stats == null) { state.stats = [:] } // library marker kkossev.commonLib, line 1352 } // library marker kkossev.commonLib, line 1353 // credits @thebearmay // library marker kkossev.commonLib, line 1355 String getModel() { // library marker kkossev.commonLib, line 1356 try { // library marker kkossev.commonLib, line 1357 /* groovylint-disable-next-line UnnecessaryGetter, UnusedVariable */ // library marker kkossev.commonLib, line 1358 String model = getHubVersion() // requires >=2.2.8.141 // library marker kkossev.commonLib, line 1359 } catch (ignore) { // library marker kkossev.commonLib, line 1360 try { // library marker kkossev.commonLib, line 1361 httpGet("http://${location.hub.localIP}:8080/api/hubitat.xml") { res -> // library marker kkossev.commonLib, line 1362 model = res.data.device.modelName // library marker kkossev.commonLib, line 1363 return model // library marker kkossev.commonLib, line 1364 } // library marker kkossev.commonLib, line 1365 } catch (ignore_again) { // library marker kkossev.commonLib, line 1366 return '' // library marker kkossev.commonLib, line 1367 } // library marker kkossev.commonLib, line 1368 } // library marker kkossev.commonLib, line 1369 } // library marker kkossev.commonLib, line 1370 // credits @thebearmay // library marker kkossev.commonLib, line 1372 boolean isCompatible(Integer minLevel) { //check to see if the hub version meets the minimum requirement ( 7 or 8 ) // library marker kkossev.commonLib, line 1373 String model = getModel() // Rev C-7 // library marker kkossev.commonLib, line 1374 String[] tokens = model.split('-') // library marker kkossev.commonLib, line 1375 String revision = tokens.last() // library marker kkossev.commonLib, line 1376 return (Integer.parseInt(revision) >= minLevel) // library marker kkossev.commonLib, line 1377 } // library marker kkossev.commonLib, line 1378 void deleteAllStatesAndJobs() { // library marker kkossev.commonLib, line 1380 state.clear() // clear all states // library marker kkossev.commonLib, line 1381 unschedule() // library marker kkossev.commonLib, line 1382 device.deleteCurrentState('*') // library marker kkossev.commonLib, line 1383 device.deleteCurrentState('') // library marker kkossev.commonLib, line 1384 log.info "${device.displayName} jobs and states cleared. HE hub is ${getHubVersion()}, version is ${location.hub.firmwareVersionString}" // library marker kkossev.commonLib, line 1386 } // library marker kkossev.commonLib, line 1387 void resetStatistics() { // library marker kkossev.commonLib, line 1389 runIn(1, 'resetStats') // library marker kkossev.commonLib, line 1390 sendInfoEvent('Statistics are reset. Refresh the web page') // library marker kkossev.commonLib, line 1391 } // library marker kkossev.commonLib, line 1392 // called from initializeVars(true) and resetStatistics() // library marker kkossev.commonLib, line 1394 void resetStats() { // library marker kkossev.commonLib, line 1395 logDebug 'resetStats...' // library marker kkossev.commonLib, line 1396 state.stats = [:] ; state.states = [:] ; state.lastRx = [:] ; state.lastTx = [:] ; state.health = [:] // library marker kkossev.commonLib, line 1397 if (this.respondsTo('groupsLibVersion')) { state.zigbeeGroups = [:] } // library marker kkossev.commonLib, line 1398 state.stats.rxCtr = 0 ; state.stats.txCtr = 0 // library marker kkossev.commonLib, line 1399 state.states['isDigital'] = false ; state.states['isRefresh'] = false ; state.states['isPing'] = false // library marker kkossev.commonLib, line 1400 state.health['offlineCtr'] = 0 ; state.health['checkCtr3'] = 0 // library marker kkossev.commonLib, line 1401 } // library marker kkossev.commonLib, line 1402 void initializeVars( boolean fullInit = false ) { // library marker kkossev.commonLib, line 1404 logDebug "InitializeVars()... fullInit = ${fullInit}" // library marker kkossev.commonLib, line 1405 if (fullInit == true ) { // library marker kkossev.commonLib, line 1406 state.clear() // library marker kkossev.commonLib, line 1407 unschedule() // library marker kkossev.commonLib, line 1408 resetStats() // library marker kkossev.commonLib, line 1409 if (deviceProfilesV3 != null && this.respondsTo('setDeviceNameAndProfile')) { setDeviceNameAndProfile() } // library marker kkossev.commonLib, line 1410 //state.comment = 'Works with Tuya Zigbee Devices' // library marker kkossev.commonLib, line 1411 logInfo 'all states and scheduled jobs cleared!' // library marker kkossev.commonLib, line 1412 state.driverVersion = driverVersionAndTimeStamp() // library marker kkossev.commonLib, line 1413 logInfo "DEVICE_TYPE = ${DEVICE_TYPE}" // library marker kkossev.commonLib, line 1414 state.deviceType = DEVICE_TYPE // library marker kkossev.commonLib, line 1415 sendInfoEvent('Initialized') // library marker kkossev.commonLib, line 1416 } // library marker kkossev.commonLib, line 1417 if (state.stats == null) { state.stats = [:] } // library marker kkossev.commonLib, line 1419 if (state.states == null) { state.states = [:] } // library marker kkossev.commonLib, line 1420 if (state.lastRx == null) { state.lastRx = [:] } // library marker kkossev.commonLib, line 1421 if (state.lastTx == null) { state.lastTx = [:] } // library marker kkossev.commonLib, line 1422 if (state.health == null) { state.health = [:] } // library marker kkossev.commonLib, line 1423 if (fullInit || settings?.txtEnable == null) { device.updateSetting('txtEnable', true) } // library marker kkossev.commonLib, line 1425 if (fullInit || settings?.logEnable == null) { device.updateSetting('logEnable', DEFAULT_DEBUG_LOGGING ?: false) } // library marker kkossev.commonLib, line 1426 if (fullInit || settings?.traceEnable == null) { device.updateSetting('traceEnable', false) } // library marker kkossev.commonLib, line 1427 if (fullInit || settings?.advancedOptions == null) { device.updateSetting('advancedOptions', [value:false, type:'bool']) } // library marker kkossev.commonLib, line 1428 if (fullInit || settings?.healthCheckMethod == null) { device.updateSetting('healthCheckMethod', [value: HealthcheckMethodOpts.defaultValue.toString(), type: 'enum']) } // library marker kkossev.commonLib, line 1429 if (fullInit || settings?.healthCheckInterval == null) { device.updateSetting('healthCheckInterval', [value: HealthcheckIntervalOpts.defaultValue.toString(), type: 'enum']) } // library marker kkossev.commonLib, line 1430 if (fullInit || settings?.voltageToPercent == null) { device.updateSetting('voltageToPercent', false) } // library marker kkossev.commonLib, line 1431 if (device.currentValue('healthStatus') == null) { sendHealthStatusEvent('unknown') } // library marker kkossev.commonLib, line 1433 // common libraries initialization // library marker kkossev.commonLib, line 1435 executeCustomHandler('batteryInitializeVars', fullInit) // added 07/06/2024 // library marker kkossev.commonLib, line 1436 executeCustomHandler('motionInitializeVars', fullInit) // added 07/06/2024 // library marker kkossev.commonLib, line 1437 executeCustomHandler('groupsInitializeVars', fullInit) // library marker kkossev.commonLib, line 1438 executeCustomHandler('illuminanceInitializeVars', fullInit) // library marker kkossev.commonLib, line 1439 executeCustomHandler('onOfInitializeVars', fullInit) // library marker kkossev.commonLib, line 1440 executeCustomHandler('energyInitializeVars', fullInit) // library marker kkossev.commonLib, line 1441 // // library marker kkossev.commonLib, line 1442 executeCustomHandler('deviceProfileInitializeVars', fullInit) // must be before the other deviceProfile initialization handlers! // library marker kkossev.commonLib, line 1443 executeCustomHandler('initEventsDeviceProfile', fullInit) // added 07/06/2024 // library marker kkossev.commonLib, line 1444 // // library marker kkossev.commonLib, line 1445 // custom device driver specific initialization should be at the end // library marker kkossev.commonLib, line 1446 executeCustomHandler('customInitializeVars', fullInit) // library marker kkossev.commonLib, line 1447 executeCustomHandler('customCreateChildDevices', fullInit) // library marker kkossev.commonLib, line 1448 executeCustomHandler('customInitEvents', fullInit) // library marker kkossev.commonLib, line 1449 final String mm = device.getDataValue('model') // library marker kkossev.commonLib, line 1451 if (mm != null) { logTrace " model = ${mm}" } // library marker kkossev.commonLib, line 1452 else { logWarn ' Model not found, please re-pair the device!' } // library marker kkossev.commonLib, line 1453 final String ep = device.getEndpointId() // library marker kkossev.commonLib, line 1454 if ( ep != null) { // library marker kkossev.commonLib, line 1455 //state.destinationEP = ep // library marker kkossev.commonLib, line 1456 logTrace " destinationEP = ${ep}" // library marker kkossev.commonLib, line 1457 } // library marker kkossev.commonLib, line 1458 else { // library marker kkossev.commonLib, line 1459 logWarn ' Destination End Point not found, please re-pair the device!' // library marker kkossev.commonLib, line 1460 //state.destinationEP = "01" // fallback // library marker kkossev.commonLib, line 1461 } // library marker kkossev.commonLib, line 1462 } // library marker kkossev.commonLib, line 1463 // not used!? // library marker kkossev.commonLib, line 1465 void setDestinationEP() { // library marker kkossev.commonLib, line 1466 String ep = device.getEndpointId() // library marker kkossev.commonLib, line 1467 if (ep != null && ep != 'F2') { state.destinationEP = ep ; logDebug "setDestinationEP() destinationEP = ${state.destinationEP}" } // library marker kkossev.commonLib, line 1468 else { logWarn "setDestinationEP() Destination End Point not found or invalid(${ep}), activating the F2 bug patch!" ; state.destinationEP = '01' } // fallback EP // library marker kkossev.commonLib, line 1469 } // library marker kkossev.commonLib, line 1470 void logDebug(final String msg) { if (settings?.logEnable) { log.debug "${device.displayName} " + msg } } // library marker kkossev.commonLib, line 1472 void logInfo(final String msg) { if (settings?.txtEnable) { log.info "${device.displayName} " + msg } } // library marker kkossev.commonLib, line 1473 void logWarn(final String msg) { if (settings?.logEnable) { log.warn "${device.displayName} " + msg } } // library marker kkossev.commonLib, line 1474 void logTrace(final String msg) { if (settings?.traceEnable) { log.trace "${device.displayName} " + msg } } // library marker kkossev.commonLib, line 1475 // _DEBUG mode only // library marker kkossev.commonLib, line 1477 void getAllProperties() { // library marker kkossev.commonLib, line 1478 log.trace 'Properties:' ; device.properties.each { it -> log.debug it } // library marker kkossev.commonLib, line 1479 log.trace 'Settings:' ; settings.each { it -> log.debug "${it.key} = ${it.value}" } // https://community.hubitat.com/t/how-do-i-get-the-datatype-for-an-app-setting/104228/6?u=kkossev // library marker kkossev.commonLib, line 1480 } // library marker kkossev.commonLib, line 1481 // delete all Preferences // library marker kkossev.commonLib, line 1483 void deleteAllSettings() { // library marker kkossev.commonLib, line 1484 String preferencesDeleted = '' // library marker kkossev.commonLib, line 1485 settings.each { it -> preferencesDeleted += "${it.key} (${it.value}), " ; device.removeSetting("${it.key}") } // library marker kkossev.commonLib, line 1486 logDebug "Deleted settings: ${preferencesDeleted}" // library marker kkossev.commonLib, line 1487 logInfo 'All settings (preferences) DELETED' // library marker kkossev.commonLib, line 1488 } // library marker kkossev.commonLib, line 1489 // delete all attributes // library marker kkossev.commonLib, line 1491 void deleteAllCurrentStates() { // library marker kkossev.commonLib, line 1492 String attributesDeleted = '' // library marker kkossev.commonLib, line 1493 device.properties.supportedAttributes.each { it -> attributesDeleted += "${it}, " ; device.deleteCurrentState("$it") } // library marker kkossev.commonLib, line 1494 logDebug "Deleted attributes: ${attributesDeleted}" ; logInfo 'All current states (attributes) DELETED' // library marker kkossev.commonLib, line 1495 } // library marker kkossev.commonLib, line 1496 // delete all State Variables // library marker kkossev.commonLib, line 1498 void deleteAllStates() { // library marker kkossev.commonLib, line 1499 String stateDeleted = '' // library marker kkossev.commonLib, line 1500 state.each { it -> stateDeleted += "${it.key}, " } // library marker kkossev.commonLib, line 1501 state.clear() // library marker kkossev.commonLib, line 1502 logDebug "Deleted states: ${stateDeleted}" ; logInfo 'All States DELETED' // library marker kkossev.commonLib, line 1503 } // library marker kkossev.commonLib, line 1504 void deleteAllScheduledJobs() { // library marker kkossev.commonLib, line 1506 unschedule() ; logInfo 'All scheduled jobs DELETED' // library marker kkossev.commonLib, line 1507 } // library marker kkossev.commonLib, line 1508 void deleteAllChildDevices() { // library marker kkossev.commonLib, line 1510 getChildDevices().each { child -> log.info "${device.displayName} Deleting ${child.deviceNetworkId}" ; deleteChildDevice(child.deviceNetworkId) } // library marker kkossev.commonLib, line 1511 sendInfoEvent 'All child devices DELETED' // library marker kkossev.commonLib, line 1512 } // library marker kkossev.commonLib, line 1513 void testParse(String par) { // library marker kkossev.commonLib, line 1515 //read attr - raw: DF8D0104020A000029280A, dni: DF8D, endpoint: 01, cluster: 0402, size: 0A, attrId: 0000, encoding: 29, command: 0A, value: 280A // library marker kkossev.commonLib, line 1516 log.trace '------------------------------------------------------' // library marker kkossev.commonLib, line 1517 log.warn "testParse - START (${par})" // library marker kkossev.commonLib, line 1518 parse(par) // library marker kkossev.commonLib, line 1519 log.warn "testParse - END (${par})" // library marker kkossev.commonLib, line 1520 log.trace '------------------------------------------------------' // library marker kkossev.commonLib, line 1521 } // library marker kkossev.commonLib, line 1522 Object testJob() { // library marker kkossev.commonLib, line 1524 log.warn 'test job executed' // library marker kkossev.commonLib, line 1525 } // library marker kkossev.commonLib, line 1526 /** // library marker kkossev.commonLib, line 1528 * Calculates and returns the cron expression // library marker kkossev.commonLib, line 1529 * @param timeInSeconds interval in seconds // library marker kkossev.commonLib, line 1530 */ // library marker kkossev.commonLib, line 1531 String getCron(int timeInSeconds) { // library marker kkossev.commonLib, line 1532 //schedule("${rnd.nextInt(59)} ${rnd.nextInt(9)}/${intervalMins} * ? * * *", 'ping') // library marker kkossev.commonLib, line 1533 // TODO: runEvery1Minute runEvery5Minutes runEvery10Minutes runEvery15Minutes runEvery30Minutes runEvery1Hour runEvery3Hours // library marker kkossev.commonLib, line 1534 final Random rnd = new Random() // library marker kkossev.commonLib, line 1535 int minutes = (timeInSeconds / 60 ) as int // library marker kkossev.commonLib, line 1536 int hours = (minutes / 60 ) as int // library marker kkossev.commonLib, line 1537 if (hours > 23) { hours = 23 } // library marker kkossev.commonLib, line 1538 String cron // library marker kkossev.commonLib, line 1539 if (timeInSeconds < 60) { cron = "*/$timeInSeconds * * * * ? *" } // library marker kkossev.commonLib, line 1540 else { // library marker kkossev.commonLib, line 1541 if (minutes < 60) { cron = "${rnd.nextInt(59)} ${rnd.nextInt(9)}/$minutes * ? * *" } // library marker kkossev.commonLib, line 1542 else { cron = "${rnd.nextInt(59)} ${rnd.nextInt(59)} */$hours ? * *" } // library marker kkossev.commonLib, line 1543 } // library marker kkossev.commonLib, line 1544 return cron // library marker kkossev.commonLib, line 1545 } // library marker kkossev.commonLib, line 1546 // credits @thebearmay // library marker kkossev.commonLib, line 1548 String formatUptime() { // library marker kkossev.commonLib, line 1549 return formatTime(location.hub.uptime) // library marker kkossev.commonLib, line 1550 } // library marker kkossev.commonLib, line 1551 String formatTime(int timeInSeconds) { // library marker kkossev.commonLib, line 1553 if (timeInSeconds == null) { return UNKNOWN } // library marker kkossev.commonLib, line 1554 int days = (timeInSeconds / 86400).toInteger() // library marker kkossev.commonLib, line 1555 int hours = ((timeInSeconds % 86400) / 3600).toInteger() // library marker kkossev.commonLib, line 1556 int minutes = ((timeInSeconds % 3600) / 60).toInteger() // library marker kkossev.commonLib, line 1557 int seconds = (timeInSeconds % 60).toInteger() // library marker kkossev.commonLib, line 1558 return "${days}d ${hours}h ${minutes}m ${seconds}s" // library marker kkossev.commonLib, line 1559 } // library marker kkossev.commonLib, line 1560 boolean isTuya() { // library marker kkossev.commonLib, line 1562 if (!device) { return true } // fallback - added 04/03/2024 // library marker kkossev.commonLib, line 1563 String model = device.getDataValue('model') // library marker kkossev.commonLib, line 1564 String manufacturer = device.getDataValue('manufacturer') // library marker kkossev.commonLib, line 1565 /* groovylint-disable-next-line UnnecessaryTernaryExpression */ // library marker kkossev.commonLib, line 1566 return (model?.startsWith('TS') && manufacturer?.startsWith('_T')) ? true : false // library marker kkossev.commonLib, line 1567 } // library marker kkossev.commonLib, line 1568 void updateTuyaVersion() { // library marker kkossev.commonLib, line 1570 if (!isTuya()) { logTrace 'not Tuya' ; return } // library marker kkossev.commonLib, line 1571 final String application = device.getDataValue('application') // library marker kkossev.commonLib, line 1572 if (application != null) { // library marker kkossev.commonLib, line 1573 Integer ver // library marker kkossev.commonLib, line 1574 try { ver = zigbee.convertHexToInt(application) } // library marker kkossev.commonLib, line 1575 catch (e) { logWarn "exception caught while converting application version ${application} to tuyaVersion"; return } // library marker kkossev.commonLib, line 1576 final String str = ((ver & 0xC0) >> 6).toString() + '.' + ((ver & 0x30) >> 4).toString() + '.' + (ver & 0x0F).toString() // library marker kkossev.commonLib, line 1577 if (device.getDataValue('tuyaVersion') != str) { // library marker kkossev.commonLib, line 1578 device.updateDataValue('tuyaVersion', str) // library marker kkossev.commonLib, line 1579 logInfo "tuyaVersion set to $str" // library marker kkossev.commonLib, line 1580 } // library marker kkossev.commonLib, line 1581 } // library marker kkossev.commonLib, line 1582 } // library marker kkossev.commonLib, line 1583 boolean isAqara() { return device.getDataValue('model')?.startsWith('lumi') ?: false } // library marker kkossev.commonLib, line 1585 void updateAqaraVersion() { // library marker kkossev.commonLib, line 1587 if (!isAqara()) { logTrace 'not Aqara' ; return } // library marker kkossev.commonLib, line 1588 String application = device.getDataValue('application') // library marker kkossev.commonLib, line 1589 if (application != null) { // library marker kkossev.commonLib, line 1590 String str = '0.0.0_' + String.format('%04d', zigbee.convertHexToInt(application.take(2))) // library marker kkossev.commonLib, line 1591 if (device.getDataValue('aqaraVersion') != str) { // library marker kkossev.commonLib, line 1592 device.updateDataValue('aqaraVersion', str) // library marker kkossev.commonLib, line 1593 logInfo "aqaraVersion set to $str" // library marker kkossev.commonLib, line 1594 } // library marker kkossev.commonLib, line 1595 } // library marker kkossev.commonLib, line 1596 } // library marker kkossev.commonLib, line 1597 String unix2formattedDate(Long unixTime) { // library marker kkossev.commonLib, line 1599 try { // library marker kkossev.commonLib, line 1600 if (unixTime == null) { return null } // library marker kkossev.commonLib, line 1601 /* groovylint-disable-next-line NoJavaUtilDate */ // library marker kkossev.commonLib, line 1602 Date date = new Date(unixTime.toLong()) // library marker kkossev.commonLib, line 1603 return date.format('yyyy-MM-dd HH:mm:ss.SSS', location.timeZone) // library marker kkossev.commonLib, line 1604 } catch (e) { // library marker kkossev.commonLib, line 1605 logDebug "Error formatting date: ${e.message}. Returning current time instead." // library marker kkossev.commonLib, line 1606 return new Date().format('yyyy-MM-dd HH:mm:ss.SSS', location.timeZone) // library marker kkossev.commonLib, line 1607 } // library marker kkossev.commonLib, line 1608 } // library marker kkossev.commonLib, line 1609 Long formattedDate2unix(String formattedDate) { // library marker kkossev.commonLib, line 1611 try { // library marker kkossev.commonLib, line 1612 if (formattedDate == null) { return null } // library marker kkossev.commonLib, line 1613 Date date = Date.parse('yyyy-MM-dd HH:mm:ss.SSS', formattedDate) // library marker kkossev.commonLib, line 1614 return date.getTime() // library marker kkossev.commonLib, line 1615 } catch (e) { // library marker kkossev.commonLib, line 1616 logDebug "Error parsing formatted date: ${formattedDate}. Returning current time instead." // library marker kkossev.commonLib, line 1617 return now() // library marker kkossev.commonLib, line 1618 } // library marker kkossev.commonLib, line 1619 } // library marker kkossev.commonLib, line 1620 static String timeToHMS(final int time) { // library marker kkossev.commonLib, line 1622 int hours = (time / 3600) as int // library marker kkossev.commonLib, line 1623 int minutes = ((time % 3600) / 60) as int // library marker kkossev.commonLib, line 1624 int seconds = time % 60 // library marker kkossev.commonLib, line 1625 return "${hours}h ${minutes}m ${seconds}s" // library marker kkossev.commonLib, line 1626 } // library marker kkossev.commonLib, line 1627 // ~~~~~ end include (144) kkossev.commonLib ~~~~~