diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/HubIOStream.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/HubIOStream.java
index 2c31660bc32..0b9d48d8ce9 100644
--- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/HubIOStream.java
+++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/HubIOStream.java
@@ -16,7 +16,6 @@ import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
-import java.nio.ByteBuffer;
import java.nio.charset.StandardCharsets;
import java.util.Base64;
import java.util.Objects;
@@ -34,8 +33,6 @@ import org.eclipse.jetty.client.api.Request;
import org.eclipse.jetty.http.HttpHeader;
import org.eclipse.jetty.http.HttpStatus;
import org.openhab.binding.insteon.internal.utils.HexUtils;
-import org.slf4j.Logger;
-import org.slf4j.LoggerFactory;
/**
* Implements IOStream for an Insteon Hub 2
@@ -47,10 +44,8 @@ import org.slf4j.LoggerFactory;
*/
@NonNullByDefault
public class HubIOStream extends IOStream {
- private final Logger logger = LoggerFactory.getLogger(HubIOStream.class);
-
- private static final String BS_START = "";
- private static final String BS_END = "";
+ private static final String BUFFER_TAG_START = "";
+ private static final String BUFFER_TAG_END = "";
private static final int REQUEST_TIMEOUT = 30; // in seconds
private String host;
@@ -61,7 +56,7 @@ public class HubIOStream extends IOStream {
private ScheduledExecutorService scheduler;
private @Nullable ScheduledFuture> job;
// index of the last byte we have read in the buffer
- private int bufferIdx = -1;
+ private volatile int bufferIdx = -1;
/**
* Constructor
@@ -84,14 +79,9 @@ public class HubIOStream extends IOStream {
this.scheduler = scheduler;
}
- @Override
- public boolean isOpen() {
- return job != null;
- }
-
@Override
public boolean open() {
- if (isOpen()) {
+ if (job != null) {
logger.warn("hub stream is already open");
return false;
}
@@ -120,93 +110,65 @@ public class HubIOStream extends IOStream {
@Override
public void close() {
+ super.close();
+
ScheduledFuture> job = this.job;
if (job != null) {
job.cancel(true);
this.job = null;
}
-
- InputStream in = this.in;
- if (in != null) {
- try {
- in.close();
- } catch (IOException e) {
- logger.debug("failed to close input stream", e);
- }
- this.in = null;
- }
-
- OutputStream out = this.out;
- if (out != null) {
- try {
- out.close();
- } catch (IOException e) {
- logger.debug("failed to close output stream", e);
- }
- this.out = null;
- }
}
/**
- * Fetches the latest status buffer from the Hub
+ * Returns the latest buffer from the Hub
*
- * @return string with status buffer
+ * @return the buffer string
* @throws IOException
*/
- private synchronized String bufferStatus() throws IOException {
+ private String getBuffer() throws IOException {
String result = getURL("/buffstatus.xml");
- int start = result.indexOf(BS_START);
- if (start == -1) {
- throw new IOException("malformed bufferstatus.xml");
- }
- start += BS_START.length();
-
- int end = result.indexOf(BS_END, start);
- if (end == -1) {
- throw new IOException("malformed bufferstatus.xml");
+ int start = result.indexOf(BUFFER_TAG_START);
+ int end = result.indexOf(BUFFER_TAG_END, start);
+ if (start == -1 || end == -1) {
+ throw new IOException("malformed buffstatus.xml");
}
+ start += BUFFER_TAG_START.length();
return result.substring(start, end).trim();
}
/**
- * Sends command to Hub to clear the status buffer
+ * Clears the Hub buffer
*
* @throws IOException
*/
- private synchronized void clearBuffer() throws IOException {
+ private void clearBuffer() throws IOException {
logger.trace("clearing buffer");
getURL("/1?XB=M=1");
bufferIdx = 0;
}
/**
- * Sends Insteon message (byte array) as a readable ascii string to the Hub
+ * Sends a message to the Hub
*
- * @param msg byte array representing the Insteon message
- * @throws IOException in case of I/O error
+ * @param b byte array representing the Insteon message
+ * @throws IOException
*/
- public synchronized void write(ByteBuffer msg) throws IOException {
- poll(); // fetch the status buffer before we send out commands
-
- StringBuilder b = new StringBuilder();
- while (msg.remaining() > 0) {
- b.append(String.format("%02x", msg.get()));
- }
- String hexMsg = b.toString();
- logger.trace("writing a message");
- getURL("/3?" + hexMsg + "=I=3");
+ private void sendMessage(byte[] b) throws IOException {
+ poll(); // poll the status buffer before we send the message
+ logger.trace("sending a message");
+ getURL("/3?" + HexUtils.getHexString(b) + "=I=3");
bufferIdx = 0;
}
/**
- * Polls the Hub web interface to fetch the status buffer
+ * Polls the Hub buffer and add to input stream
*
- * @throws IOException if something goes wrong with I/O
+ * @throws IOException
*/
- private synchronized void poll() throws IOException {
- String buffer = bufferStatus(); // fetch via http call
+ private void poll() throws IOException {
+ String buffer = getBuffer();
logger.trace("poll: {}", buffer);
// The Hub maintains a ring buffer where the last two digits (in hex!) represent
// the position of the last byte read.
@@ -249,10 +211,9 @@ public class HubIOStream extends IOStream {
logger.trace("no wrap: appending new data: {}", msg);
}
if (msg.length() != 0) {
- byte[] array = HexUtils.toByteArray(msg.toString());
- ByteBuffer buf = ByteBuffer.wrap(array);
+ byte[] b = HexUtils.toByteArray(msg.toString());
if (in instanceof HubInputStream hubInput) {
- hubInput.handle(buf);
+ hubInput.add(b);
} else {
logger.debug("hub input stream is null");
}
@@ -310,10 +271,10 @@ public class HubIOStream extends IOStream {
// A buffer to keep bytes while we are waiting for the inputstream to read
private ReadByteBuffer buffer = new ReadByteBuffer(1024);
- public void handle(ByteBuffer b) throws IOException {
+ public void add(byte[] b) throws IOException {
// Make sure we cleanup as much space as possible
buffer.makeCompact();
- buffer.add(b.array());
+ buffer.add(b);
}
@Override
@@ -342,18 +303,18 @@ public class HubIOStream extends IOStream {
@Override
public void write(int b) throws IOException {
out.write(b);
- flushBuffer();
+ flush();
}
@Override
public void write(byte @Nullable [] b, int off, int len) throws IOException {
out.write(b, off, len);
- flushBuffer();
+ flush();
}
- private void flushBuffer() throws IOException {
- ByteBuffer buffer = ByteBuffer.wrap(out.toByteArray());
- HubIOStream.this.write(buffer);
+ @Override
+ public void flush() throws IOException {
+ sendMessage(out.toByteArray());
out.reset();
}
}
diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/IOStream.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/IOStream.java
index 1e2cb90646d..a51a3ac69f0 100644
--- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/IOStream.java
+++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/IOStream.java
@@ -26,6 +26,8 @@ import org.openhab.binding.insteon.internal.config.InsteonHub1Configuration;
import org.openhab.binding.insteon.internal.config.InsteonHub2Configuration;
import org.openhab.binding.insteon.internal.config.InsteonPLMConfiguration;
import org.openhab.core.io.transport.serial.SerialPortManager;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
/**
* Abstract class for implementation for I/O stream with anything that looks
@@ -38,6 +40,7 @@ import org.openhab.core.io.transport.serial.SerialPortManager;
*/
@NonNullByDefault
public abstract class IOStream {
+ protected final Logger logger = LoggerFactory.getLogger(getClass());
protected @Nullable InputStream in;
protected @Nullable OutputStream out;
@@ -47,20 +50,17 @@ public abstract class IOStream {
*
* @param b byte array (output)
* @return number of bytes read
+ * @throws InterruptedException
+ * @throws IOException
*/
public int read(byte @Nullable [] b) throws InterruptedException, IOException {
+ InputStream in = this.in;
+ if (in == null) {
+ throw new IOException("input stream not defined");
+ }
int len = 0;
while (len == 0) {
- if (!isOpen()) {
- throw new IOException("io stream not open");
- }
-
- InputStream in = this.in;
- if (in != null) {
- len = in.read(b);
- } else {
- throw new IOException("input stream not defined");
- }
+ len = in.read(b);
if (Thread.interrupted()) {
throw new InterruptedException();
@@ -77,27 +77,16 @@ public abstract class IOStream {
* Writes data to IOStream
*
* @param b byte array to write
+ * @throws IOException
*/
- public void write(byte @Nullable [] b) throws InterruptedException, IOException {
- if (!isOpen()) {
- throw new IOException("io stream not open");
- }
-
+ public void write(byte @Nullable [] b) throws IOException {
OutputStream out = this.out;
- if (out != null) {
- out.write(b);
- } else {
+ if (out == null) {
throw new IOException("output stream not defined");
}
+ out.write(b);
}
- /**
- * Returns if IOStream is open
- *
- * @return true if stream is open, false if not
- */
- public abstract boolean isOpen();
-
/**
* Opens the IOStream
*
@@ -108,7 +97,27 @@ public abstract class IOStream {
/**
* Closes the IOStream
*/
- public abstract void close();
+ public void close() {
+ InputStream in = this.in;
+ if (in != null) {
+ try {
+ in.close();
+ } catch (IOException e) {
+ logger.debug("failed to close input stream", e);
+ }
+ this.in = null;
+ }
+
+ OutputStream out = this.out;
+ if (out != null) {
+ try {
+ out.close();
+ } catch (IOException e) {
+ logger.debug("failed to close output stream", e);
+ }
+ this.out = null;
+ }
+ }
/**
* Creates an IOStream from an insteon bridge config object
diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/LegacyPort.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/LegacyPort.java
index 6d8ba6166db..4ed258a9179 100644
--- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/LegacyPort.java
+++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/LegacyPort.java
@@ -205,9 +205,7 @@ public class LegacyPort {
mdbb.stop();
}
- if (ioStream.isOpen()) {
- ioStream.close();
- }
+ ioStream.close();
ScheduledFuture> readJob = this.readJob;
if (readJob != null) {
diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/Port.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/Port.java
index f57d15da7c9..ad51cadd732 100644
--- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/Port.java
+++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/Port.java
@@ -135,9 +135,7 @@ public class Port {
connected.set(false);
- if (ioStream.isOpen()) {
- ioStream.close();
- }
+ ioStream.close();
ScheduledFuture> readJob = this.readJob;
if (readJob != null) {
diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/SerialIOStream.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/SerialIOStream.java
index c7eef6348fa..42959860218 100644
--- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/SerialIOStream.java
+++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/SerialIOStream.java
@@ -13,8 +13,6 @@
package org.openhab.binding.insteon.internal.transport;
import java.io.IOException;
-import java.io.InputStream;
-import java.io.OutputStream;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
@@ -24,8 +22,6 @@ import org.openhab.core.io.transport.serial.SerialPort;
import org.openhab.core.io.transport.serial.SerialPortIdentifier;
import org.openhab.core.io.transport.serial.SerialPortManager;
import org.openhab.core.io.transport.serial.UnsupportedCommOperationException;
-import org.slf4j.Logger;
-import org.slf4j.LoggerFactory;
/**
* Implements IOStream for serial devices
@@ -37,8 +33,6 @@ import org.slf4j.LoggerFactory;
*/
@NonNullByDefault
public class SerialIOStream extends IOStream {
- private final Logger logger = LoggerFactory.getLogger(SerialIOStream.class);
-
private String name;
private int baudRate;
private SerialPortManager serialPortManager;
@@ -50,14 +44,9 @@ public class SerialIOStream extends IOStream {
this.serialPortManager = serialPortManager;
}
- @Override
- public boolean isOpen() {
- return port != null;
- }
-
@Override
public boolean open() {
- if (isOpen()) {
+ if (port != null) {
logger.warn("serial port is already open");
return false;
}
@@ -93,25 +82,7 @@ public class SerialIOStream extends IOStream {
@Override
public void close() {
- InputStream in = this.in;
- if (in != null) {
- try {
- in.close();
- } catch (IOException e) {
- logger.debug("failed to close input stream", e);
- }
- this.in = null;
- }
-
- OutputStream out = this.out;
- if (out != null) {
- try {
- out.close();
- } catch (IOException e) {
- logger.debug("failed to close output stream", e);
- }
- this.out = null;
- }
+ super.close();
SerialPort port = this.port;
if (port != null) {
diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/TcpIOStream.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/TcpIOStream.java
index 9f6bcb57859..06bbda45c2b 100644
--- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/TcpIOStream.java
+++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/TcpIOStream.java
@@ -13,15 +13,11 @@
package org.openhab.binding.insteon.internal.transport;
import java.io.IOException;
-import java.io.InputStream;
-import java.io.OutputStream;
import java.net.Socket;
import java.net.UnknownHostException;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
-import org.slf4j.Logger;
-import org.slf4j.LoggerFactory;
/**
* Implements IOStream for an Insteon Legacy Hub
@@ -33,8 +29,6 @@ import org.slf4j.LoggerFactory;
*/
@NonNullByDefault
public class TcpIOStream extends IOStream {
- private final Logger logger = LoggerFactory.getLogger(TcpIOStream.class);
-
private String host;
private int port;
private @Nullable Socket socket;
@@ -50,14 +44,9 @@ public class TcpIOStream extends IOStream {
this.port = port;
}
- @Override
- public boolean isOpen() {
- return socket != null;
- }
-
@Override
public boolean open() {
- if (isOpen()) {
+ if (socket != null) {
logger.warn("socket is already open");
return false;
}
@@ -79,25 +68,7 @@ public class TcpIOStream extends IOStream {
@Override
public void close() {
- InputStream in = this.in;
- if (in != null) {
- try {
- in.close();
- } catch (IOException e) {
- logger.debug("failed to close input stream", e);
- }
- this.in = null;
- }
-
- OutputStream out = this.out;
- if (out != null) {
- try {
- out.close();
- } catch (IOException e) {
- logger.debug("failed to close output stream", e);
- }
- this.out = null;
- }
+ super.close();
Socket socket = this.socket;
if (socket != null) {
diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/message/MsgFactory.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/message/MsgFactory.java
index 0555e04cce3..ba9e50ac7ce 100644
--- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/message/MsgFactory.java
+++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/transport/message/MsgFactory.java
@@ -78,7 +78,7 @@ public class MsgFactory {
end += l;
// copy the incoming data to the end of the buffer
if (logger.isTraceEnabled()) {
- logger.trace("read buffer: len {} data: {}", end, HexUtils.getHexString(buf, end, false));
+ logger.trace("read buffer: len {} data: {}", end, HexUtils.getHexString(buf, end));
}
}
@@ -136,7 +136,7 @@ public class MsgFactory {
done = true;
}
if (logger.isTraceEnabled()) {
- logger.trace("keeping buffer len {} data: {}", end, HexUtils.getHexString(buf, end, false));
+ logger.trace("keeping buffer len {} data: {}", end, HexUtils.getHexString(buf, end));
}
return msg;
}
diff --git a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/utils/HexUtils.java b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/utils/HexUtils.java
index dbb2b1fa29b..cb7b7b43026 100644
--- a/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/utils/HexUtils.java
+++ b/bundles/org.openhab.binding.insteon/src/main/java/org/openhab/binding/insteon/internal/utils/HexUtils.java
@@ -71,33 +71,29 @@ public class HexUtils {
return s;
}
+ /**
+ * Returns a hex string for a given byte array
+ *
+ * @param bytes the byte array
+ * @return the formatted hex string
+ */
+ public static String getHexString(byte[] bytes) {
+ return getHexString(bytes, bytes.length);
+ }
+
/**
* Returns a hex string for a given byte array and length
*
* @param bytes the byte array
* @param len the string length
* @return the formatted hex string
+ * @throws ArrayIndexOutOfBoundsException
*/
- public static String getHexString(byte[] bytes, int len) {
- return getHexString(bytes, len, true);
- }
-
- /**
- * Returns a hex string for a given byte array, length and prefix flag
- *
- * @param bytes the byte array
- * @param len the string length
- * @param addPrefix if hex prefix should be added
- * @return the formatted hex string
- */
- public static String getHexString(byte[] bytes, int len, boolean addPrefix) {
+ public static String getHexString(byte[] bytes, int len) throws ArrayIndexOutOfBoundsException {
String s = "";
- for (int i = 0; i < bytes.length && i < len; i++) {
+ for (int i = 0; i < len; i++) {
s += String.format("%02X", bytes[i] & 0xFF);
}
- if (!s.isEmpty() && addPrefix) {
- s = "0x" + s;
- }
return s;
}