Skip to content

Commit

Permalink
Update Instantaneous Demand channel to use floating point (#668)
Browse files Browse the repository at this point in the history
Signed-off-by: Chris Jackson <[email protected]>
  • Loading branch information
cdjackson authored Aug 12, 2021
1 parent a6bd07e commit 1c0d8f9
Show file tree
Hide file tree
Showing 2 changed files with 92 additions and 8 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -48,8 +48,8 @@ public class ZigBeeConverterMeteringInstantaneousDemand extends ZigBeeBaseChanne

private ZclAttribute attribute;

private Integer divisor;
private Integer multiplier;
private double divisor = 1.0;
private double multiplier = 1.0;

@Override
public Set<Integer> getImplementedClientClusters() {
Expand Down Expand Up @@ -160,7 +160,7 @@ public void attributeUpdated(ZclAttribute attribute, Object val) {
logger.debug("{}: ZigBee attribute reports {}", endpoint.getIeeeAddress(), attribute);
if (attribute.getCluster() == ZclClusterType.METERING
&& attribute.getId() == ZclMeteringCluster.ATTR_INSTANTANEOUSDEMAND) {
Integer value = (Integer) val;
double value = (Integer) val;
BigDecimal valueCalibrated = BigDecimal.valueOf(value * multiplier / divisor);
updateChannelState(new DecimalType(valueCalibrated));
}
Expand All @@ -170,12 +170,15 @@ private void determineDivisorAndMultiplier(ZclMeteringCluster serverClusterMeasu
ZclAttribute divisorAttribute = clusterMetering.getAttribute(ZclMeteringCluster.ATTR_DIVISOR);
ZclAttribute multiplierAttribute = clusterMetering.getAttribute(ZclMeteringCluster.ATTR_MULTIPLIER);

divisor = (Integer) divisorAttribute.readValue(Long.MAX_VALUE);
multiplier = (Integer) multiplierAttribute.readValue(Long.MAX_VALUE);
if (divisor == null || multiplier == null) {
divisor = 1;
multiplier = 1;
Integer iDiv = (Integer) divisorAttribute.readValue(Long.MAX_VALUE);
Integer iMult = (Integer) multiplierAttribute.readValue(Long.MAX_VALUE);
if (iDiv == null || iMult == null) {
iDiv = 1;
iMult = 1;
}

divisor = iDiv;
multiplier = iMult;
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
/**
* Copyright (c) 2010-2021 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.zigbee.internal.converter;

import static org.junit.jupiter.api.Assertions.assertEquals;

import org.junit.jupiter.api.Test;
import org.mockito.ArgumentCaptor;
import org.mockito.ArgumentMatchers;
import org.mockito.Mockito;
import org.openhab.binding.zigbee.handler.ZigBeeCoordinatorHandler;
import org.openhab.binding.zigbee.handler.ZigBeeThingHandler;
import org.openhab.core.library.types.DecimalType;
import org.openhab.core.thing.Channel;
import org.openhab.core.thing.ChannelUID;
import org.openhab.core.thing.binding.builder.ChannelBuilder;
import org.openhab.core.types.State;

import com.zsmartsystems.zigbee.IeeeAddress;
import com.zsmartsystems.zigbee.ZigBeeEndpoint;
import com.zsmartsystems.zigbee.zcl.ZclAttribute;
import com.zsmartsystems.zigbee.zcl.clusters.ZclMeteringCluster;
import com.zsmartsystems.zigbee.zcl.protocol.ZclDataType;

/**
* Tests for metering converter
*
* @author Chris Jackson
*
*/
public class ZigBeeConverterMeteringInstantaneousDemandTest {

@Test
public void testAttributeUpdated() {
ZigBeeEndpoint endpoint = Mockito.mock(ZigBeeEndpoint.class);
ZigBeeCoordinatorHandler coordinatorHandler = Mockito.mock(ZigBeeCoordinatorHandler.class);
ZclMeteringCluster cluster = Mockito.mock(ZclMeteringCluster.class);
ZclAttribute divAttribute = Mockito.mock(ZclAttribute.class);
ZclAttribute multAttribute = Mockito.mock(ZclAttribute.class);
ZclAttribute valueAttribute = Mockito.mock(ZclAttribute.class);
Mockito.when(cluster.getAttribute(ZclMeteringCluster.ATTR_INSTANTANEOUSDEMAND)).thenReturn(valueAttribute);
Mockito.when(cluster.getAttribute(ZclMeteringCluster.ATTR_DIVISOR)).thenReturn(divAttribute);
Mockito.when(cluster.getAttribute(ZclMeteringCluster.ATTR_MULTIPLIER)).thenReturn(multAttribute);
Mockito.when(divAttribute.readValue(Long.MAX_VALUE)).thenReturn(10000);
Mockito.when(multAttribute.readValue(Long.MAX_VALUE)).thenReturn(1);
Mockito.when(valueAttribute.readValue(Long.MAX_VALUE)).thenReturn(65);

Mockito.when(coordinatorHandler.getEndpoint(ArgumentMatchers.any(IeeeAddress.class), ArgumentMatchers.anyInt()))
.thenReturn(endpoint);
Mockito.when(endpoint.getInputCluster(ZclMeteringCluster.CLUSTER_ID)).thenReturn(cluster);

ZigBeeConverterMeteringInstantaneousDemand converter = new ZigBeeConverterMeteringInstantaneousDemand();
ArgumentCaptor<ChannelUID> channelCapture = ArgumentCaptor.forClass(ChannelUID.class);
ArgumentCaptor<State> stateCapture = ArgumentCaptor.forClass(State.class);
ZigBeeThingHandler thingHandler = Mockito.mock(ZigBeeThingHandler.class);
Channel channel = ChannelBuilder.create(new ChannelUID("a:b:c:d"), "").build();
converter.initialize(channel, coordinatorHandler, new IeeeAddress("1234567890ABCDEF"), 1);
converter.initializeConverter(thingHandler);

ZclAttribute attribute = new ZclAttribute(new ZclMeteringCluster(endpoint),
ZclMeteringCluster.ATTR_INSTANTANEOUSDEMAND, "Demand", ZclDataType.SIGNED_24_BIT_INTEGER, false, false,
false, false);

attribute.updateValue(65);
converter.attributeUpdated(attribute, attribute.getLastValue());
Mockito.verify(thingHandler, Mockito.times(1)).setChannelState(channelCapture.capture(),
stateCapture.capture());
assertEquals(new ChannelUID("a:b:c:d"), channelCapture.getValue());
assertEquals(DecimalType.valueOf("0.0065"), stateCapture.getValue());
}
}

0 comments on commit 1c0d8f9

Please sign in to comment.