]> git.basschouten.com Git - openhab-addons.git/blob
b1f2dd41c5220c29390b148191107efc7ab0245a
[openhab-addons.git] /
1 /**
2  * Copyright (c) 2010-2020 Contributors to the openHAB project
3  *
4  * See the NOTICE file(s) distributed with this work for additional
5  * information.
6  *
7  * This program and the accompanying materials are made available under the
8  * terms of the Eclipse Public License 2.0 which is available at
9  * http://www.eclipse.org/legal/epl-2.0
10  *
11  * SPDX-License-Identifier: EPL-2.0
12  */
13 package org.openhab.binding.dsmr.internal.discovery;
14
15 import java.util.*;
16 import java.util.Map.Entry;
17 import java.util.stream.Collectors;
18
19 import org.eclipse.jdt.annotation.NonNullByDefault;
20 import org.openhab.binding.dsmr.internal.device.cosem.CosemObject;
21 import org.openhab.binding.dsmr.internal.device.cosem.CosemObjectType;
22 import org.openhab.binding.dsmr.internal.device.p1telegram.P1Telegram;
23 import org.openhab.binding.dsmr.internal.device.p1telegram.P1TelegramListener;
24 import org.openhab.binding.dsmr.internal.handler.DSMRBridgeHandler;
25 import org.openhab.binding.dsmr.internal.handler.DSMRMeterHandler;
26 import org.openhab.binding.dsmr.internal.meter.DSMRMeterDescriptor;
27 import org.openhab.binding.dsmr.internal.meter.DSMRMeterType;
28 import org.openhab.core.i18n.LocaleProvider;
29 import org.openhab.core.i18n.TranslationProvider;
30 import org.openhab.core.library.types.StringType;
31 import org.openhab.core.thing.Thing;
32 import org.slf4j.Logger;
33 import org.slf4j.LoggerFactory;
34
35 /**
36  * This implements the discovery service for new DSMR Meters on a active DSMR bridge.
37  *
38  * @author M. Volaart - Initial contribution
39  * @author Hilbrand Bouwkamp - Refactored code to detect meters during actual discovery phase.
40  */
41 @NonNullByDefault
42 public class DSMRMeterDiscoveryService extends DSMRDiscoveryService implements P1TelegramListener {
43
44     private final Logger logger = LoggerFactory.getLogger(DSMRMeterDiscoveryService.class);
45
46     /**
47      * The {@link DSMRBridgeHandler} instance
48      */
49     private final DSMRBridgeHandler dsmrBridgeHandler;
50
51     /**
52      * Constructs a new {@link DSMRMeterDiscoveryService} attached to the give bridge handler.
53      *
54      * @param dsmrBridgeHandler The bridge handler this discovery service is attached to
55      */
56     public DSMRMeterDiscoveryService(DSMRBridgeHandler dsmrBridgeHandler) {
57         this.dsmrBridgeHandler = dsmrBridgeHandler;
58     }
59
60     @Override
61     protected void startScan() {
62         logger.debug("Start discovery on existing DSMR bridge.");
63         dsmrBridgeHandler.setLenientMode(true);
64         dsmrBridgeHandler.registerDSMRMeterListener(this);
65     }
66
67     @Override
68     protected synchronized void stopScan() {
69         logger.debug("Stop discovery on existing DSMR bridge.");
70         dsmrBridgeHandler.setLenientMode(false);
71         super.stopScan();
72         dsmrBridgeHandler.unregisterDSMRMeterListener(this);
73     }
74
75     @Override
76     public void telegramReceived(P1Telegram telegram) {
77         if (logger.isDebugEnabled()) {
78             logger.debug("Detect meters from #{} objects", telegram.getCosemObjects().size());
79         }
80         final Entry<Collection<DSMRMeterDescriptor>, Map<CosemObjectType, CosemObject>> detectedMeters = meterDetector
81                 .detectMeters(telegram);
82         verifyUnregisteredCosemObjects(telegram, detectedMeters.getValue());
83         validateConfiguredMeters(dsmrBridgeHandler.getThing().getThings(),
84                 detectedMeters.getKey().stream().map(md -> md.getMeterType()).collect(Collectors.toSet()));
85         detectedMeters.getKey().forEach(m -> meterDiscovered(m, dsmrBridgeHandler.getThing().getUID()));
86     }
87
88     protected void verifyUnregisteredCosemObjects(P1Telegram telegram,
89             Map<CosemObjectType, CosemObject> undetectedCosemObjects) {
90         if (!undetectedCosemObjects.isEmpty()) {
91             if (undetectedCosemObjects.entrySet().stream()
92                     .anyMatch(e -> e.getKey() == CosemObjectType.METER_EQUIPMENT_IDENTIFIER
93                             && e.getValue().getCosemValues().entrySet().stream().anyMatch(
94                                     cv -> cv.getValue() instanceof StringType && cv.getValue().toString().isEmpty()))) {
95                 // Unregistered meter detected. log to the user.
96                 reportUnregisteredMeters();
97             } else {
98                 reportUnrecognizedCosemObjects(undetectedCosemObjects);
99                 logger.info("There are unrecognized cosem values in the data received from the meter,"
100                         + " which means some meters might not be detected. Please report your raw data as reference: {}",
101                         telegram.getRawTelegram());
102             }
103         }
104         if (!telegram.getUnknownCosemObjects().isEmpty()) {
105             logger.info("There are unrecognized cosem values in the data received from the meter,"
106                     + " which means you have values that can't be read by a channel: {}. Please report them and your raw data as reference: {}",
107                     telegram.getUnknownCosemObjects().stream()
108                             .map(e -> String.format("obis id:%s, value:%s", e.getKey(), e.getValue()))
109                             .collect(Collectors.joining(", ")),
110                     telegram.getRawTelegram());
111         }
112     }
113
114     /**
115      * Called when Unrecognized cosem objects where found. This can be a bug or a new meter not yet supported.
116      *
117      * @param unidentifiedCosemObjects Map with the unrecognized.
118      */
119     protected void reportUnrecognizedCosemObjects(Map<CosemObjectType, CosemObject> unidentifiedCosemObjects) {
120         unidentifiedCosemObjects
121                 .forEach((k, v) -> logger.info("Unrecognized cosem object '{}' found in the data: {}", k, v));
122     }
123
124     /**
125      * Called when a meter equipment identifier is found that has an empty value. This
126      */
127     protected void reportUnregisteredMeters() {
128         logger.info(
129                 "An unregistered meter has been found. Probably a new meter. Retry discovery once the meter is registered with the energy provider.");
130     }
131
132     /**
133      * Validates if the meters configured by the user match with what is detected in the telegram. Some meters are a
134      * subset of other meters and therefore an invalid configured meter does work, but not all available data is
135      * available to the user.
136      *
137      * @param things The list of configured things
138      * @param configuredMeterTypes The set of meters detected in the telegram
139      */
140     private void validateConfiguredMeters(List<Thing> things, Set<DSMRMeterType> configuredMeterTypes) {
141         // @formatter:off
142         final Set<DSMRMeterType> configuredMeters = things.stream()
143                 .map(Thing::getHandler)
144                 .filter(DSMRMeterHandler.class::isInstance)
145                 .map(DSMRMeterHandler.class::cast)
146                 .map(h -> h == null ? null : h.getMeterDescriptor())
147                 .map(d -> Optional.ofNullable(d == null ? null : d.getMeterType()))
148                 .filter(Optional::isPresent)
149                 .map(Optional::get)
150                 .collect(Collectors.toSet());
151         // @formatter:on
152         // Create list of all configured meters that are not in the detected list. If not empty meters might not be
153         // correctly configured.
154         final List<DSMRMeterType> invalidConfigured = configuredMeters.stream()
155                 .filter(dm -> !configuredMeterTypes.contains(dm)).collect(Collectors.toList());
156         // Create a list of all detected meters not yet configured.
157         final List<DSMRMeterType> unconfiguredMeters = configuredMeterTypes.stream()
158                 .filter(dm -> !configuredMeters.contains(dm)).collect(Collectors.toList());
159
160         if (!invalidConfigured.isEmpty()) {
161             reportConfigurationValidationResults(invalidConfigured, unconfiguredMeters);
162         }
163     }
164
165     /**
166      * Called when the validation finds in inconsistency between configured meters.
167      *
168      * @param invalidConfigured The list of invalid configured meters
169      * @param unconfiguredMeters The list of meters that were detected, but not configured
170      */
171     protected void reportConfigurationValidationResults(List<DSMRMeterType> invalidConfigured,
172             List<DSMRMeterType> unconfiguredMeters) {
173         logger.info(
174                 "Possible incorrect meters configured. These are configured: {}."
175                         + "But the following unconfigured meters are found in the data received from the meter: {}",
176                 invalidConfigured.stream().map(m -> m.name()).collect(Collectors.joining(", ")),
177                 unconfiguredMeters.stream().map(m -> m.name()).collect(Collectors.joining(", ")));
178     }
179
180     public void setLocaleProvider(final LocaleProvider localeProvider) {
181         this.localeProvider = localeProvider;
182     }
183
184     public void unsetLocaleProvider() {
185         this.localeProvider = null;
186     }
187
188     public void setTranslationProvider(TranslationProvider i18nProvider) {
189         this.i18nProvider = i18nProvider;
190     }
191
192     public void unsetTranslationProvider() {
193         this.i18nProvider = null;
194     }
195 }