2 * Copyright (c) 2010-2024 Contributors to the openHAB project
4 * See the NOTICE file(s) distributed with this work for additional
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
11 * SPDX-License-Identifier: EPL-2.0
13 package org.openhab.binding.insteon.internal.device;
15 import java.io.IOException;
16 import java.util.List;
18 import java.util.Map.Entry;
19 import java.util.concurrent.ScheduledExecutorService;
20 import java.util.concurrent.ScheduledFuture;
21 import java.util.concurrent.TimeUnit;
23 import org.eclipse.jdt.annotation.NonNullByDefault;
24 import org.eclipse.jdt.annotation.Nullable;
25 import org.openhab.binding.insteon.internal.driver.ModemDBEntry;
26 import org.openhab.binding.insteon.internal.driver.Port;
27 import org.openhab.binding.insteon.internal.message.FieldException;
28 import org.openhab.binding.insteon.internal.message.InvalidMessageTypeException;
29 import org.openhab.binding.insteon.internal.message.Msg;
30 import org.openhab.binding.insteon.internal.message.MsgListener;
31 import org.openhab.binding.insteon.internal.utils.Utils;
32 import org.slf4j.Logger;
33 import org.slf4j.LoggerFactory;
36 * Builds the modem database from incoming link record messages
38 * @author Bernd Pfrommer - Initial contribution
39 * @author Rob Nielsen - Port to openHAB 2 insteon binding
42 public class ModemDBBuilder implements MsgListener {
43 private static final int MESSAGE_TIMEOUT = 30000;
45 private final Logger logger = LoggerFactory.getLogger(ModemDBBuilder.class);
47 private volatile boolean isComplete = false;
49 private ScheduledExecutorService scheduler;
50 private @Nullable ScheduledFuture<?> job = null;
51 private volatile long lastMessageTimestamp;
52 private volatile int messageCount = 0;
54 public ModemDBBuilder(Port port, ScheduledExecutorService scheduler) {
56 this.scheduler = scheduler;
60 port.addListener(this);
62 logger.trace("starting modem db builder");
64 job = scheduler.scheduleWithFixedDelay(() -> {
66 logger.trace("modem db builder finished");
67 ScheduledFuture<?> job = this.job;
73 if (System.currentTimeMillis() - lastMessageTimestamp > MESSAGE_TIMEOUT) {
75 if (messageCount == 0) {
77 No messages were received, the PLM or hub might be broken. If this continues see \
78 'Known Limitations and Issues' in the Insteon binding documentation.\
81 logger.warn("Modem database download was unsuccessful, restarting!{}", s);
85 }, 0, 1, TimeUnit.SECONDS);
88 private void startDownload() {
89 logger.trace("starting modem database download");
91 lastMessageTimestamp = System.currentTimeMillis();
96 public boolean isComplete() {
100 private void getFirstLinkRecord() {
102 port.writeMessage(Msg.makeMessage("GetFirstALLLinkRecord"));
103 } catch (IOException e) {
104 logger.warn("error sending link record query ", e);
105 } catch (InvalidMessageTypeException e) {
106 logger.warn("invalid message ", e);
111 * processes link record messages from the modem to build database
112 * and request more link records if not finished.
116 public void msg(Msg msg) {
117 lastMessageTimestamp = System.currentTimeMillis();
120 if (msg.isPureNack()) {
124 if (msg.getByte("Cmd") == 0x69 || msg.getByte("Cmd") == 0x6a) {
125 // If the flag is "ACK/NACK", a record response
126 // will follow, so we do nothing here.
127 // If its "NACK", there are none
128 if (msg.getByte("ACK/NACK") == 0x15) {
129 logger.debug("got all link records.");
132 } else if (msg.getByte("Cmd") == 0x57) {
133 // we got the link record response
134 updateModemDB(msg.getAddress("LinkAddr"), port, msg, false);
135 port.writeMessage(Msg.makeMessage("GetNextALLLinkRecord"));
137 } catch (FieldException e) {
138 logger.debug("bad field handling link records {}", e.getMessage());
139 } catch (IOException e) {
140 logger.debug("got IO exception handling link records {}", e.getMessage());
141 } catch (IllegalStateException e) {
142 logger.debug("got exception requesting link records {}", e.getMessage());
143 } catch (InvalidMessageTypeException e) {
144 logger.warn("invalid message ", e);
148 private synchronized void done() {
151 port.removeListener(this);
152 port.modemDBComplete();
155 private void logModemDB() {
157 logger.debug("MDB ------- start of modem link records ------------------");
158 Map<InsteonAddress, ModemDBEntry> dbes = port.getDriver().lockModemDBEntries();
159 for (Entry<InsteonAddress, ModemDBEntry> db : dbes.entrySet()) {
160 List<Msg> lrs = db.getValue().getLinkRecords();
162 int recordFlags = m.getByte("RecordFlags") & 0xff;
163 String ms = ((recordFlags & (0x1 << 6)) != 0) ? "CTRL" : "RESP";
164 logger.debug("MDB {}: {} group: {} data1: {} data2: {} data3: {}", db.getKey(), ms,
165 toHex(m.getByte("ALLLinkGroup")), toHex(m.getByte("LinkData1")),
166 toHex(m.getByte("LinkData2")), toHex(m.getByte("LinkData2")));
168 logger.debug("MDB -----");
170 logger.debug("MDB ---------------- end of modem link records -----------");
171 } catch (FieldException e) {
172 logger.warn("cannot access field:", e);
174 port.getDriver().unlockModemDBEntries();
178 public static String toHex(byte b) {
179 return Utils.getHexString(b);
182 public void updateModemDB(InsteonAddress linkAddr, Port port, @Nullable Msg m, boolean isModem) {
184 Map<InsteonAddress, ModemDBEntry> dbes = port.getDriver().lockModemDBEntries();
185 ModemDBEntry dbe = dbes.get(linkAddr);
187 dbe = new ModemDBEntry(linkAddr, isModem);
188 dbes.put(linkAddr, dbe);
192 dbe.addLinkRecord(m);
194 byte group = m.getByte("ALLLinkGroup");
195 int recordFlags = m.getByte("RecordFlags") & 0xff;
196 if ((recordFlags & (0x1 << 6)) != 0) {
197 dbe.addControls(group);
199 dbe.addRespondsTo(group);
201 } catch (FieldException e) {
202 logger.warn("cannot access field:", e);
206 port.getDriver().unlockModemDBEntries();