From d54a51b73263853d5a387e34937e0b824bb24a8f Mon Sep 17 00:00:00 2001 From: Jeremy Date: Sat, 4 Jan 2025 15:10:41 -0500 Subject: [PATCH] [insteon] Add modem database backup restore console commands (#17958) Signed-off-by: jsetton --- .../internal/InsteonBindingConstants.java | 4 +- .../internal/command/DebugCommand.java | 55 +++---- .../internal/command/InsteonCommand.java | 21 +++ .../internal/command/ModemCommand.java | 143 +++++++++++++++++- .../internal/command/SceneCommand.java | 1 - .../internal/device/InsteonDevice.java | 2 +- .../insteon/internal/device/InsteonModem.java | 38 +++-- .../device/database/DatabaseRecord.java | 14 +- .../internal/device/database/LinkDB.java | 8 +- .../device/database/LinkDBReader.java | 4 +- .../device/database/LinkDBWriter.java | 4 +- .../internal/device/database/ModemDB.java | 9 ++ .../device/database/ModemDBReader.java | 7 - .../device/database/ModemDBRecord.java | 35 ++++- .../device/database/ModemDBWriter.java | 2 +- 15 files changed, 266 insertions(+), 81 deletions(-) diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/InsteonBindingConstants.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/InsteonBindingConstants.java index 449cf717142..3f375fbff7f 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/InsteonBindingConstants.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/InsteonBindingConstants.java @@ -12,7 +12,7 @@ */ package org.openhab.binding.insteon.internal; -import java.io.File; +import java.nio.file.Path; import java.util.List; import java.util.Map; import java.util.Set; @@ -34,7 +34,7 @@ import org.openhab.core.thing.ThingTypeUID; @NonNullByDefault public class InsteonBindingConstants { public static final String BINDING_ID = "insteon"; - public static final String BINDING_DATA_DIR = OpenHAB.getUserDataFolder() + File.separator + BINDING_ID; + public static final Path BINDING_DATA_DIR = Path.of(OpenHAB.getUserDataFolder(), BINDING_ID); // List of all thing type uids public static final ThingTypeUID THING_TYPE_DEVICE = new ThingTypeUID(BINDING_ID, "device"); diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/DebugCommand.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/DebugCommand.java index 6c3604dcf90..8a029fe79ff 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/DebugCommand.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/DebugCommand.java @@ -13,11 +13,11 @@ package org.openhab.binding.insteon.internal.command; import java.io.File; -import java.io.FileOutputStream; import java.io.IOException; -import java.io.PrintStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.StandardOpenOption; import java.text.SimpleDateFormat; -import java.util.Arrays; import java.util.Date; import java.util.HashSet; import java.util.List; @@ -27,7 +27,6 @@ import java.util.stream.Collectors; import java.util.stream.Stream; import org.eclipse.jdt.annotation.NonNullByDefault; -import org.openhab.binding.insteon.internal.InsteonBindingConstants; import org.openhab.binding.insteon.internal.device.InsteonAddress; import org.openhab.binding.insteon.internal.device.InsteonScene; import org.openhab.binding.insteon.internal.device.X10Address; @@ -71,7 +70,7 @@ public class DebugCommand extends InsteonCommand implements PortListener { private static final String ALL_OPTION = "--all"; - private static final String MSG_EVENTS_FILE_PREFIX = "messageEvents"; + private static final String MSG_EVENTS_FILE_PREFIX = "message-events"; private static enum MessageType { STANDARD, @@ -195,9 +194,16 @@ public class DebugCommand extends InsteonCommand implements PortListener { } else if (cursorArgumentIndex == 1) { switch (args[0]) { case START_MONITORING: + strings = monitorAllDevices ? List.of() + : Stream.concat(Stream.of(ALL_OPTION), + getModem().getDB().getDevices().stream() + .filter(address -> !monitoredAddresses.contains(address)) + .map(InsteonAddress::toString)) + .toList(); + break; case STOP_MONITORING: - strings = Stream.concat(Stream.of(ALL_OPTION), - getModem().getDB().getDevices().stream().map(InsteonAddress::toString)).toList(); + strings = monitorAllDevices ? List.of(ALL_OPTION) + : monitoredAddresses.stream().map(InsteonAddress::toString).toList(); break; case SEND_BROADCAST_MESSAGE: strings = getModem().getDB().getBroadcastGroups().stream().map(String::valueOf).toList(); @@ -251,40 +257,25 @@ public class DebugCommand extends InsteonCommand implements PortListener { } } - private String getMsgEventsFileName(String address) { - return MSG_EVENTS_FILE_PREFIX + "-" + address.replace(".", "") + ".log"; - } - - private String getMsgEventsFilePath(String address) { - return InsteonBindingConstants.BINDING_DATA_DIR + File.separator + getMsgEventsFileName(address); + private Path getMsgEventsFilePath(String address) { + return getBindingDataFilePath(MSG_EVENTS_FILE_PREFIX + "-" + address.toLowerCase().replace(".", "") + ".log"); } private void clearMonitorFiles(String address) { - File folder = new File(InsteonBindingConstants.BINDING_DATA_DIR); - String prefix = ALL_OPTION.equals(address) ? MSG_EVENTS_FILE_PREFIX : getMsgEventsFileName(address); + String prefix = ALL_OPTION.equals(address) ? MSG_EVENTS_FILE_PREFIX + : getMsgEventsFilePath(address).getFileName().toString(); - if (folder.isDirectory()) { - Arrays.asList(folder.listFiles()).stream().filter(file -> file.getName().startsWith(prefix)) - .forEach(File::delete); - } + getBindingDataFilePaths(prefix).map(Path::toFile).forEach(File::delete); } private void logMessageEvent(InsteonAddress address, Msg msg) { String timestamp = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS").format(new Date()); - String pathname = getMsgEventsFilePath(address.toString()); + String line = timestamp + " " + msg + System.lineSeparator(); + Path path = getMsgEventsFilePath(address.toString()); try { - File file = new File(pathname); - File parent = file.getParentFile(); - if (parent == null) { - throw new IOException(pathname + " does not name a parent directory"); - } - parent.mkdirs(); - file.createNewFile(); - - PrintStream ps = new PrintStream(new FileOutputStream(file, true)); - ps.println(timestamp + " " + msg.toString()); - ps.close(); + Files.createDirectories(path.getParent()); + Files.writeString(path, line, StandardOpenOption.CREATE, StandardOpenOption.APPEND); } catch (IOException e) { logger.warn("failed to write to message event file", e); } @@ -307,7 +298,7 @@ public class DebugCommand extends InsteonCommand implements PortListener { monitorAllDevices = true; monitoredAddresses.clear(); console.println("Started monitoring all devices."); - console.println("Message events logged in " + InsteonBindingConstants.BINDING_DATA_DIR); + console.println("Message events logged in " + getBindingDataDirPath()); clearMonitorFiles(address); } else { console.println("Already monitoring all devices."); diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/InsteonCommand.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/InsteonCommand.java index 8857ea6dbfa..e9bfc5cf91e 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/InsteonCommand.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/InsteonCommand.java @@ -12,6 +12,9 @@ */ package org.openhab.binding.insteon.internal.command; +import java.io.IOException; +import java.nio.file.Files; +import java.nio.file.Path; import java.util.List; import java.util.Map; import java.util.Map.Entry; @@ -20,6 +23,7 @@ import java.util.stream.Stream; import org.eclipse.jdt.annotation.NonNullByDefault; import org.eclipse.jdt.annotation.Nullable; +import org.openhab.binding.insteon.internal.InsteonBindingConstants; import org.openhab.binding.insteon.internal.device.Device; import org.openhab.binding.insteon.internal.device.InsteonAddress; import org.openhab.binding.insteon.internal.device.InsteonDevice; @@ -171,4 +175,21 @@ public abstract class InsteonCommand implements ConsoleCommandCompleter { InsteonDevice device = getInsteonDevice(thingId); return device != null ? device.getInsteonEngine() : InsteonEngine.UNKNOWN; } + + protected Path getBindingDataDirPath() { + return InsteonBindingConstants.BINDING_DATA_DIR; + } + + protected Path getBindingDataFilePath(String filename) { + return InsteonBindingConstants.BINDING_DATA_DIR.resolve(filename); + } + + protected Stream getBindingDataFilePaths(String prefix) { + try { + return Files.list(InsteonBindingConstants.BINDING_DATA_DIR) + .filter(path -> path.getFileName().toString().startsWith(prefix)); + } catch (IOException e) { + return Stream.of(); + } + } } diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/ModemCommand.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/ModemCommand.java index c6a282ccedd..006c682a7b9 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/ModemCommand.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/ModemCommand.java @@ -12,6 +12,12 @@ */ package org.openhab.binding.insteon.internal.command; +import java.io.IOException; +import java.io.InputStream; +import java.nio.file.Files; +import java.nio.file.Path; +import java.text.SimpleDateFormat; +import java.util.Date; import java.util.List; import java.util.Map; import java.util.stream.Collectors; @@ -20,6 +26,7 @@ import org.eclipse.jdt.annotation.NonNullByDefault; import org.eclipse.jdt.annotation.Nullable; import org.openhab.binding.insteon.internal.device.InsteonAddress; import org.openhab.binding.insteon.internal.device.ProductData; +import org.openhab.binding.insteon.internal.device.database.ModemDBChange; import org.openhab.binding.insteon.internal.device.database.ModemDBEntry; import org.openhab.binding.insteon.internal.device.database.ModemDBRecord; import org.openhab.binding.insteon.internal.handler.InsteonBridgeHandler; @@ -41,6 +48,8 @@ public class ModemCommand extends InsteonCommand { private static final String LIST_ALL = "listAll"; private static final String LIST_DATABASE = "listDatabase"; private static final String RELOAD_DATABASE = "reloadDatabase"; + private static final String BACKUP_DATABASE = "backupDatabase"; + private static final String RESTORE_DATABASE = "restoreDatabase"; private static final String ADD_DATABASE_CONTROLLER = "addDatabaseController"; private static final String ADD_DATABASE_RESPONDER = "addDatabaseResponder"; private static final String DELETE_DATABASE_RECORD = "deleteDatabaseRecord"; @@ -48,16 +57,19 @@ public class ModemCommand extends InsteonCommand { private static final String CLEAR_DATABASE_CHANGES = "clearDatabaseChanges"; private static final String ADD_DEVICE = "addDevice"; private static final String REMOVE_DEVICE = "removeDevice"; + private static final String RESET = "reset"; private static final String SWITCH = "switch"; - private static final List SUBCMDS = List.of(LIST_ALL, LIST_DATABASE, RELOAD_DATABASE, - ADD_DATABASE_CONTROLLER, ADD_DATABASE_RESPONDER, DELETE_DATABASE_RECORD, APPLY_DATABASE_CHANGES, - CLEAR_DATABASE_CHANGES, ADD_DEVICE, REMOVE_DEVICE, SWITCH); + private static final List SUBCMDS = List.of(LIST_ALL, LIST_DATABASE, RELOAD_DATABASE, BACKUP_DATABASE, + RESTORE_DATABASE, ADD_DATABASE_CONTROLLER, ADD_DATABASE_RESPONDER, DELETE_DATABASE_RECORD, + APPLY_DATABASE_CHANGES, CLEAR_DATABASE_CHANGES, ADD_DEVICE, REMOVE_DEVICE, RESET, SWITCH); private static final String CONFIRM_OPTION = "--confirm"; private static final String FORCE_OPTION = "--force"; private static final String RECORDS_OPTION = "--records"; + private static final String MODEM_DATABASE_FILE_PREFIX = "modem-database"; + public ModemCommand(InsteonCommandExtension commandExtension) { super(NAME, DESCRIPTION, commandExtension); } @@ -69,6 +81,9 @@ public class ModemCommand extends InsteonCommand { buildCommandUsage(LIST_DATABASE + " [" + RECORDS_OPTION + "]", "list all-link database summary or records and pending changes for the Insteon modem"), buildCommandUsage(RELOAD_DATABASE, "reload all-link database from the Insteon modem"), + buildCommandUsage(BACKUP_DATABASE, "backup all-link database from the Insteon modem to a file"), + buildCommandUsage(RESTORE_DATABASE + " " + CONFIRM_OPTION, + "restore all-link database to the Insteon modem from a specific file"), buildCommandUsage(ADD_DATABASE_CONTROLLER + "
[ ]", "add a controller record to all-link database for the Insteon modem"), buildCommandUsage(ADD_DATABASE_RESPONDER + "
", @@ -83,6 +98,7 @@ public class ModemCommand extends InsteonCommand { "add an Insteon device to the modem, optionally providing its address"), buildCommandUsage(REMOVE_DEVICE + "
[" + FORCE_OPTION + "]", "remove an Insteon device from the modem"), + buildCommandUsage(RESET + " " + CONFIRM_OPTION, "reset the Insteon modem to factory defaults"), buildCommandUsage(SWITCH + " ", "switch Insteon modem bridge to use if more than one configured and enabled")); } @@ -118,6 +134,20 @@ public class ModemCommand extends InsteonCommand { printUsage(console, args[0]); } break; + case BACKUP_DATABASE: + if (args.length == 1) { + backupDatabase(console); + } else { + printUsage(console, args[0]); + } + break; + case RESTORE_DATABASE: + if (args.length == 2 || args.length == 3 && CONFIRM_OPTION.equals(args[2])) { + restoreDatabase(console, args[1], args.length == 3); + } else { + printUsage(console, args[0]); + } + break; case ADD_DATABASE_CONTROLLER: if (args.length == 3 || args.length == 6) { addDatabaseRecord(console, args, true); @@ -167,6 +197,13 @@ public class ModemCommand extends InsteonCommand { printUsage(console, args[0]); } break; + case RESET: + if (args.length == 1 || args.length == 2 && CONFIRM_OPTION.equals(args[1])) { + resetModem(console, args.length == 2); + } else { + printUsage(console, args[0]); + } + break; case SWITCH: if (args.length == 2) { switchModem(console, args[1]); @@ -191,6 +228,10 @@ public class ModemCommand extends InsteonCommand { case LIST_DATABASE: strings = List.of(RECORDS_OPTION); break; + case RESTORE_DATABASE: + strings = getBindingDataFilePaths(MODEM_DATABASE_FILE_PREFIX).map(Path::getFileName) + .map(Path::toString).toList(); + break; case ADD_DATABASE_CONTROLLER: case ADD_DATABASE_RESPONDER: case REMOVE_DEVICE: @@ -200,6 +241,10 @@ public class ModemCommand extends InsteonCommand { strings = getModem().getDB().getRecords().stream().map(record -> record.getAddress().toString()) .distinct().toList(); break; + case APPLY_DATABASE_CHANGES: + case RESET: + strings = List.of(CONFIRM_OPTION); + break; case SWITCH: strings = getBridgeHandlers().map(InsteonBridgeHandler::getThingId).toList(); break; @@ -207,6 +252,9 @@ public class ModemCommand extends InsteonCommand { } else if (cursorArgumentIndex == 2) { InsteonAddress address = InsteonAddress.isValid(args[1]) ? new InsteonAddress(args[1]) : null; switch (args[0]) { + case RESTORE_DATABASE: + strings = List.of(CONFIRM_OPTION); + break; case DELETE_DATABASE_RECORD: if (address != null) { strings = getModem().getDB().getRecords(address).stream() @@ -242,7 +290,8 @@ public class ModemCommand extends InsteonCommand { } else if (entries.isEmpty()) { console.println("The all-link database for modem " + address + " is empty"); } else { - console.println("The all-link database for modem " + address + " contains " + entries.size() + " devices:"); + console.println("The all-link database for modem " + address + " contains " + entries.size() + " devices:" + + (!getModem().getDB().isComplete() ? " (Partial)" : "")); print(console, entries); } } @@ -255,7 +304,8 @@ public class ModemCommand extends InsteonCommand { } else if (records.isEmpty()) { console.println("The all-link database for modem " + address + " is empty"); } else { - console.println("The all-link database for modem " + address + " contains " + records.size() + " records:"); + console.println("The all-link database for modem " + address + " contains " + records.size() + " records:" + + (!getModem().getDB().isComplete() ? " (Partial)" : "")); print(console, records); listDatabaseChanges(console); } @@ -263,7 +313,7 @@ public class ModemCommand extends InsteonCommand { private void listDatabaseChanges(Console console) { InsteonAddress address = getModem().getAddress(); - List changes = getModem().getDB().getChanges().stream().map(String::valueOf).toList(); + List changes = getModem().getDB().getChanges().stream().map(ModemDBChange::toString).toList(); if (InsteonAddress.UNKNOWN.equals(address)) { console.println("No modem found!"); } else if (!changes.isEmpty()) { @@ -285,6 +335,73 @@ public class ModemCommand extends InsteonCommand { } } + private void backupDatabase(Console console) { + InsteonAddress address = getModem().getAddress(); + if (InsteonAddress.UNKNOWN.equals(address)) { + console.println("No modem found!"); + } else if (!getModem().getDB().isComplete()) { + console.println("The all-link database for modem " + address + " is not loaded yet."); + } else if (getModem().getDB().getRecords().isEmpty()) { + console.println("The all-link database for modem " + address + " is empty"); + } else { + String timestamp = new SimpleDateFormat("yyyyMMdd-HHmmss").format(new Date()); + String id = address.toString().toLowerCase().replace(".", ""); + Path path = getBindingDataFilePath(MODEM_DATABASE_FILE_PREFIX + "-" + id + "-" + timestamp + ".dmp"); + byte[] bytes = getModem().getDB().getRecordDump(); + int count = bytes.length / ModemDBRecord.SIZE; + + try { + Files.createDirectories(path.getParent()); + Files.write(path, bytes); + console.println("Backed up " + count + " database records from modem " + address + " to " + path); + } catch (IOException e) { + console.println("Failed to write backup file: " + e.getMessage()); + } + } + } + + private void restoreDatabase(Console console, String filename, boolean isConfirmed) { + InsteonAddress address = getModem().getAddress(); + if (InsteonAddress.UNKNOWN.equals(address)) { + console.println("No modem found!"); + } else if (!getModem().getDB().isComplete()) { + console.println("The all-link database for modem " + address + " is not loaded yet."); + } else { + Path path = Path.of(filename); + if (!path.isAbsolute()) { + path = getBindingDataFilePath(filename); + } + + try { + if (!Files.isReadable(path)) { + console.println("The restore file " + path + " does not exist or is not readable."); + } else if (Files.size(path) == 0) { + console.println("The restore file " + path + " is empty."); + } else { + InputStream stream = Files.newInputStream(path); + List records = ModemDBRecord.fromRecordDump(stream); + if (!isConfirmed) { + console.println( + "The restore file " + path + " contains " + records.size() + " database records:"); + print(console, records.stream().map(ModemDBRecord::toString).toList()); + console.println("Please run the same command with " + CONFIRM_OPTION + + " option to have these database records restored to modem " + address + "."); + } else { + console.println("Restoring " + records.size() + " database records to modem " + address + + " from " + path + "..."); + records.forEach(record -> getModem().getDB().markRecordForAddOrModify(record.getAddress(), + record.getGroup(), record.isController(), record.getData())); + getModem().getDB().update(); + } + } + } catch (IllegalArgumentException e) { + console.println("The restore file " + path + " is invalid."); + } catch (IOException e) { + console.println("Failed to read restore file: " + e.getMessage()); + } + } + } + private void addDatabaseRecord(Console console, String[] args, boolean isController) { if (!getModem().getDB().isComplete()) { console.println("The modem database is not loaded yet."); @@ -313,7 +430,6 @@ public class ModemCommand extends InsteonCommand { ModemDBRecord record = getModem().getDB().getRecord(address, group, isController); if (record == null) { getModem().getDB().markRecordForAdd(address, group, isController, data); - } else { getModem().getDB().markRecordForModify(record, data); } @@ -405,6 +521,19 @@ public class ModemCommand extends InsteonCommand { } } + private void resetModem(Console console, boolean isConfirmed) { + InsteonAddress address = getModem().getAddress(); + if (InsteonAddress.UNKNOWN.equals(address)) { + console.println("No modem found!"); + } else if (!isConfirmed) { + console.println("Please run the same command with " + CONFIRM_OPTION + " option to reset modem " + address + + " to factory defaults."); + } else { + console.println("Resetting modem " + address + " to factory defaults..."); + getModem().reset(); + } + } + private void switchModem(Console console, String thingId) { InsteonBridgeHandler handler = getBridgeHandler(thingId); if (handler == null) { diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/SceneCommand.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/SceneCommand.java index 6630f0dad04..ad39d1c46c9 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/SceneCommand.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/command/SceneCommand.java @@ -153,7 +153,6 @@ public class SceneCommand extends InsteonCommand { } break; } - } else if (cursorArgumentIndex == 4) { InsteonDevice device = getInsteonDevice(args[2]); DeviceFeature feature = device != null ? device.getFeature(args[3]) : null; diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/InsteonDevice.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/InsteonDevice.java index 2d788f77f0c..5b62dd134e9 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/InsteonDevice.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/InsteonDevice.java @@ -980,7 +980,7 @@ public class InsteonDevice extends BaseDevice IntStream.range(0, bytes.length).map(i -> bytes[i])) + .collect(ByteArrayOutputStream::new, ByteArrayOutputStream::write, + (out1, out2) -> out1.write(out2.toByteArray(), 0, out2.size())) + .toByteArray(); + } + private Stream getRecords(@Nullable InsteonAddress address, @Nullable Integer group, @Nullable Boolean isController) { return getRecords().stream() diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBReader.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBReader.java index 62a1b9c19bd..61b441d3f7e 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBReader.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBReader.java @@ -175,9 +175,6 @@ public class ModemDBReader implements PortListener { } else if (msg.getCommand() == 0x53) { // we got a linking completed message handleLinkingCompleted(msg); - } else if (msg.getCommand() == 0x55 || msg.getCommand() == 0x67 && msg.isReplyAck()) { - // we got a user reset detected message or im reset reply ack - handleIMReset(); } else if (msg.getCommand() == 0x57) { // we got a link record response handleLinkRecord(msg); @@ -305,8 +302,4 @@ public class ModemDBReader implements PortListener { productQueries.remove(address); } } - - private void handleIMReset() { - modem.resetInitiated(); - } } diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBRecord.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBRecord.java index 066d01bb26d..be91c41b58e 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBRecord.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBRecord.java @@ -12,6 +12,11 @@ */ package org.openhab.binding.insteon.internal.device.database; +import java.io.IOException; +import java.io.InputStream; +import java.util.ArrayList; +import java.util.List; + import org.eclipse.jdt.annotation.NonNullByDefault; import org.openhab.binding.insteon.internal.device.InsteonAddress; import org.openhab.binding.insteon.internal.transport.message.FieldException; @@ -26,7 +31,7 @@ import org.openhab.binding.insteon.internal.transport.message.Msg; public class ModemDBRecord extends DatabaseRecord { public ModemDBRecord(RecordType type, int group, InsteonAddress address, byte[] data) { - super(LOCATION_ZERO, type, group, address, data); + super(type, group, address, data); } public ModemDBRecord(DatabaseRecord record) { @@ -99,6 +104,34 @@ public class ModemDBRecord extends DatabaseRecord { return new ModemDBRecord(type, group, address, data); } + /** + * Factory method for creating a list of ModemDBRecord from an Insteon record dump + * + * @param stream the Insteon record dump input stream to use + * @return the list of modem db records + * @throws IllegalArgumentException + * @throws IOException + */ + public static List fromRecordDump(InputStream stream) throws IllegalArgumentException, IOException { + List records = new ArrayList<>(); + + if (stream.available() % ModemDBRecord.SIZE != 0) { + throw new IllegalArgumentException("Invalid record dump length"); + } + + while (stream.available() > 0) { + byte[] buf = stream.readNBytes(ModemDBRecord.SIZE); + RecordType type = new RecordType(Byte.toUnsignedInt(buf[0])); + int group = Byte.toUnsignedInt(buf[1]); + InsteonAddress address = new InsteonAddress(buf[2], buf[3], buf[4]); + byte[] data = new byte[] { buf[5], buf[6], buf[7] }; + + records.add(new ModemDBRecord(type, group, address, data)); + } + + return records; + } + /** * Factory method for creating a new ModemDBRecord from another instance with new data * diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBWriter.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBWriter.java index 5cca46c8f01..348e3c44452 100644 --- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBWriter.java +++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/device/database/ModemDBWriter.java @@ -94,7 +94,7 @@ public class ModemDBWriter implements PortListener { private void manageNextModemLinkRecord() { ModemDBChange change = modem.getDB().pollNextChange(); if (change == null) { - logger.trace("all modem database changes written"); + logger.debug("all modem database changes written"); done(); } else { ModemDBRecord record = change.getRecord();