diff --git a/bom/openhab-core/pom.xml b/bom/openhab-core/pom.xml
index 00c4a1dff15..ad505b5ef4e 100644
--- a/bom/openhab-core/pom.xml
+++ b/bom/openhab-core/pom.xml
@@ -316,6 +316,12 @@
${project.version}
compile
+
+ org.openhab.core.bundles
+ org.openhab.core.config.discovery.usbserial.ser2net
+ ${project.version}
+ compile
+
org.openhab.core.bundles
org.openhab.core.config.discovery.upnp
diff --git a/bundles/org.openhab.core.config.discovery.usbserial.ser2net/.classpath b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/.classpath
new file mode 100644
index 00000000000..4244343f8aa
--- /dev/null
+++ b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/.classpath
@@ -0,0 +1,29 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/bundles/org.openhab.core.config.discovery.usbserial.ser2net/.project b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/.project
new file mode 100644
index 00000000000..30e9c110128
--- /dev/null
+++ b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/.project
@@ -0,0 +1,23 @@
+
+
+ org.openhab.core.config.discovery.usbserial.ser2net
+
+
+
+
+
+ org.eclipse.jdt.core.javabuilder
+
+
+
+
+ org.eclipse.m2e.core.maven2Builder
+
+
+
+
+
+ org.eclipse.jdt.core.javanature
+ org.eclipse.m2e.core.maven2Nature
+
+
diff --git a/bundles/org.openhab.core.config.discovery.usbserial.ser2net/NOTICE b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/NOTICE
new file mode 100644
index 00000000000..6c17d0d8a45
--- /dev/null
+++ b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/NOTICE
@@ -0,0 +1,14 @@
+This content is produced and maintained by the openHAB project.
+
+* Project home: https://www.openhab.org
+
+== Declared Project Licenses
+
+This program and the accompanying materials are made available under the terms
+of the Eclipse Public License 2.0 which is available at
+https://www.eclipse.org/legal/epl-2.0/.
+
+== Source Code
+
+/~https://github.com/openhab/openhab-core
+
diff --git a/bundles/org.openhab.core.config.discovery.usbserial.ser2net/pom.xml b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/pom.xml
new file mode 100644
index 00000000000..1384c0fd078
--- /dev/null
+++ b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/pom.xml
@@ -0,0 +1,30 @@
+
+
+
+ 4.0.0
+
+
+ org.openhab.core.bundles
+ org.openhab.core.reactor.bundles
+ 3.2.0-SNAPSHOT
+
+
+ org.openhab.core.config.discovery.usbserial.ser2net
+
+ openHAB Core :: Bundles :: Configuration USB-Serial Discovery using ser2net mDNS scanning
+
+
+
+ org.openhab.core.bundles
+ org.openhab.core.config.discovery.usbserial
+ ${project.version}
+
+
+ org.openhab.core.bundles
+ org.openhab.core.io.transport.mdns
+ ${project.version}
+
+
+
+
diff --git a/bundles/org.openhab.core.config.discovery.usbserial.ser2net/src/main/java/org/openhab/core/config/discovery/usbserial/ser2net/internal/Ser2NetUsbSerialDiscovery.java b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/src/main/java/org/openhab/core/config/discovery/usbserial/ser2net/internal/Ser2NetUsbSerialDiscovery.java
new file mode 100644
index 00000000000..4b7d2ac7c84
--- /dev/null
+++ b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/src/main/java/org/openhab/core/config/discovery/usbserial/ser2net/internal/Ser2NetUsbSerialDiscovery.java
@@ -0,0 +1,206 @@
+/**
+ * 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.core.config.discovery.usbserial.ser2net.internal;
+
+import java.time.Duration;
+import java.util.HashSet;
+import java.util.Optional;
+import java.util.Set;
+import java.util.concurrent.CopyOnWriteArraySet;
+import java.util.stream.Collectors;
+import java.util.stream.Stream;
+
+import javax.jmdns.ServiceEvent;
+import javax.jmdns.ServiceInfo;
+import javax.jmdns.ServiceListener;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDeviceInformation;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDiscovery;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDiscoveryListener;
+import org.openhab.core.io.transport.mdns.MDNSClient;
+import org.osgi.service.component.annotations.Activate;
+import org.osgi.service.component.annotations.Component;
+import org.osgi.service.component.annotations.Reference;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+
+/**
+ * A {@link UsbSerialDiscovery} that implements background discovery of RFC2217 by listening to
+ * ser2net mDNS service events.
+ *
+ * @author Wouter Born - Initial contribution
+ */
+@NonNullByDefault
+@Component(service = UsbSerialDiscovery.class)
+public class Ser2NetUsbSerialDiscovery implements ServiceListener, UsbSerialDiscovery {
+
+ private final Logger logger = LoggerFactory.getLogger(Ser2NetUsbSerialDiscovery.class);
+
+ static final String SERVICE_TYPE = "_iostream._tcp.local.";
+
+ static final String PROPERTY_PROVIDER = "provider";
+ static final String PROPERTY_DEVICE_TYPE = "devicetype";
+ static final String PROPERTY_GENSIO_STACK = "gensiostack";
+
+ static final String PROPERTY_VENDOR_ID = "idVendor";
+ static final String PROPERTY_PRODUCT_ID = "idProduct";
+
+ static final String PROPERTY_SERIAL_NUMBER = "serial";
+ static final String PROPERTY_MANUFACTURER = "manufacturer";
+ static final String PROPERTY_PRODUCT = "product";
+
+ static final String PROPERTY_INTERFACE_NUMBER = "bInterfaceNumber";
+ static final String PROPERTY_INTERFACE = "interface";
+
+ static final String SER2NET = "ser2net";
+ static final String SERIALUSB = "serialusb";
+ static final String TELNET_RFC2217_TCP = "telnet(rfc2217),tcp";
+
+ static final Duration SINGLE_SCAN_DURATION = Duration.ofSeconds(5);
+ static final String SERIAL_PORT_NAME_FORMAT = "rfc2217://%s:%s";
+
+ private final Set discoveryListeners = new CopyOnWriteArraySet<>();
+ private final MDNSClient mdnsClient;
+
+ private boolean notifyListeners = false;
+
+ private Set lastScanResult = new HashSet<>();
+
+ @Activate
+ public Ser2NetUsbSerialDiscovery(final @Reference MDNSClient mdnsClient) {
+ this.mdnsClient = mdnsClient;
+ }
+
+ @Override
+ public void registerDiscoveryListener(UsbSerialDiscoveryListener listener) {
+ discoveryListeners.add(listener);
+ }
+
+ @Override
+ public void unregisterDiscoveryListener(UsbSerialDiscoveryListener listener) {
+ discoveryListeners.remove(listener);
+ }
+
+ @Override
+ public synchronized void startBackgroundScanning() {
+ notifyListeners = true;
+ mdnsClient.addServiceListener(SERVICE_TYPE, this);
+ logger.debug("Started ser2net USB-Serial mDNS background discovery");
+ }
+
+ @Override
+ public synchronized void stopBackgroundScanning() {
+ notifyListeners = false;
+ mdnsClient.removeServiceListener(SERVICE_TYPE, this);
+ logger.debug("Stopped ser2net USB-Serial mDNS background discovery");
+ }
+
+ @Override
+ public synchronized void doSingleScan() {
+ logger.debug("Starting ser2net USB-Serial mDNS single discovery scan");
+
+ Set scanResult = Stream.of(mdnsClient.list(SERVICE_TYPE, SINGLE_SCAN_DURATION))
+ .map(this::createUsbSerialDeviceInformation) //
+ .filter(Optional::isPresent) //
+ .map(Optional::get) //
+ .collect(Collectors.toSet());
+
+ Set added = setDifference(scanResult, lastScanResult);
+ Set removed = setDifference(lastScanResult, scanResult);
+ Set unchanged = setDifference(scanResult, added);
+
+ lastScanResult = scanResult;
+
+ removed.stream().forEach(this::announceRemovedDevice);
+ added.stream().forEach(this::announceAddedDevice);
+ unchanged.stream().forEach(this::announceAddedDevice);
+
+ logger.debug("Completed ser2net USB-Serial mDNS single discovery scan");
+ }
+
+ private Set setDifference(Set set1, Set set2) {
+ Set result = new HashSet<>(set1);
+ result.removeAll(set2);
+ return result;
+ }
+
+ private void announceAddedDevice(UsbSerialDeviceInformation deviceInfo) {
+ for (UsbSerialDiscoveryListener listener : discoveryListeners) {
+ listener.usbSerialDeviceDiscovered(deviceInfo);
+ }
+ }
+
+ private void announceRemovedDevice(UsbSerialDeviceInformation deviceInfo) {
+ for (UsbSerialDiscoveryListener listener : discoveryListeners) {
+ listener.usbSerialDeviceRemoved(deviceInfo);
+ }
+ }
+
+ @Override
+ public void serviceAdded(@NonNullByDefault({}) ServiceEvent event) {
+ if (notifyListeners) {
+ Optional deviceInfo = createUsbSerialDeviceInformation(event.getInfo());
+ deviceInfo.ifPresent(this::announceAddedDevice);
+ }
+ }
+
+ @Override
+ public void serviceRemoved(@NonNullByDefault({}) ServiceEvent event) {
+ if (notifyListeners) {
+ Optional deviceInfo = createUsbSerialDeviceInformation(event.getInfo());
+ deviceInfo.ifPresent(this::announceRemovedDevice);
+ }
+ }
+
+ @Override
+ public void serviceResolved(@NonNullByDefault({}) ServiceEvent event) {
+ serviceAdded(event);
+ }
+
+ private Optional createUsbSerialDeviceInformation(ServiceInfo serviceInfo) {
+ String provider = serviceInfo.getPropertyString(PROPERTY_PROVIDER);
+ String deviceType = serviceInfo.getPropertyString(PROPERTY_DEVICE_TYPE);
+ String gensioStack = serviceInfo.getPropertyString(PROPERTY_GENSIO_STACK);
+
+ // Check ser2net specific properties when present
+ if (SER2NET.equals(provider) && (deviceType != null && !SERIALUSB.equals(deviceType))
+ || (gensioStack != null && !TELNET_RFC2217_TCP.equals(gensioStack))) {
+ logger.debug("Skipping creation of UsbSerialDeviceInformation based on {}", serviceInfo);
+ return Optional.empty();
+ }
+
+ try {
+ int vendorId = Integer.parseInt(serviceInfo.getPropertyString(PROPERTY_VENDOR_ID), 16);
+ int productId = Integer.parseInt(serviceInfo.getPropertyString(PROPERTY_PRODUCT_ID), 16);
+
+ String serialNumber = serviceInfo.getPropertyString(PROPERTY_SERIAL_NUMBER);
+ String manufacturer = serviceInfo.getPropertyString(PROPERTY_MANUFACTURER);
+ String product = serviceInfo.getPropertyString(PROPERTY_PRODUCT);
+
+ int interfaceNumber = Integer.parseInt(serviceInfo.getPropertyString(PROPERTY_INTERFACE_NUMBER), 16);
+ String interfaceDescription = serviceInfo.getPropertyString(PROPERTY_INTERFACE);
+
+ String serialPortName = String.format(SERIAL_PORT_NAME_FORMAT, serviceInfo.getHostAddresses()[0],
+ serviceInfo.getPort());
+
+ UsbSerialDeviceInformation deviceInfo = new UsbSerialDeviceInformation(vendorId, productId, serialNumber,
+ manufacturer, product, interfaceNumber, interfaceDescription, serialPortName);
+ logger.debug("Created {} based on {}", deviceInfo, serviceInfo);
+ return Optional.of(deviceInfo);
+ } catch (NumberFormatException e) {
+ logger.debug("Failed to create UsbSerialDeviceInformation based on {}", serviceInfo, e);
+ return Optional.empty();
+ }
+ }
+}
diff --git a/bundles/org.openhab.core.config.discovery.usbserial.ser2net/src/test/java/org/openhab/core/config/discovery/usbserial/ser2net/internal/Ser2NetUsbSerialDiscoveryTest.java b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/src/test/java/org/openhab/core/config/discovery/usbserial/ser2net/internal/Ser2NetUsbSerialDiscoveryTest.java
new file mode 100644
index 00000000000..6b28f2be411
--- /dev/null
+++ b/bundles/org.openhab.core.config.discovery.usbserial.ser2net/src/test/java/org/openhab/core/config/discovery/usbserial/ser2net/internal/Ser2NetUsbSerialDiscoveryTest.java
@@ -0,0 +1,269 @@
+/**
+ * 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.core.config.discovery.usbserial.ser2net.internal;
+
+import static org.mockito.ArgumentMatchers.*;
+import static org.mockito.Mockito.*;
+import static org.openhab.core.config.discovery.usbserial.ser2net.internal.Ser2NetUsbSerialDiscovery.*;
+
+import java.io.IOException;
+import java.time.Duration;
+
+import javax.jmdns.ServiceEvent;
+import javax.jmdns.ServiceInfo;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.ArgumentMatchers;
+import org.mockito.Mock;
+import org.mockito.junit.jupiter.MockitoExtension;
+import org.mockito.junit.jupiter.MockitoSettings;
+import org.mockito.quality.Strictness;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDeviceInformation;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDiscoveryListener;
+import org.openhab.core.io.transport.mdns.MDNSClient;
+
+/**
+ * Unit tests for the {@link Ser2NetUsbSerialDiscovery}.
+ *
+ * @author Wouter Born - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+@MockitoSettings(strictness = Strictness.LENIENT)
+public class Ser2NetUsbSerialDiscoveryTest {
+
+ private @Mock @NonNullByDefault({}) UsbSerialDiscoveryListener discoveryListenerMock;
+ private @Mock @NonNullByDefault({}) MDNSClient mdnsClientMock;
+
+ private @Mock @NonNullByDefault({}) ServiceInfo serviceInfo1Mock;
+ private @Mock @NonNullByDefault({}) ServiceInfo serviceInfo2Mock;
+ private @Mock @NonNullByDefault({}) ServiceInfo serviceInfo3Mock;
+ private @Mock @NonNullByDefault({}) ServiceInfo invalidServiceInfoMock;
+
+ private @Mock @NonNullByDefault({}) ServiceEvent serviceEvent1Mock;
+ private @Mock @NonNullByDefault({}) ServiceEvent serviceEvent2Mock;
+ private @Mock @NonNullByDefault({}) ServiceEvent serviceEvent3Mock;
+ private @Mock @NonNullByDefault({}) ServiceEvent invalidServiceEventMock;
+
+ private @NonNullByDefault({}) Ser2NetUsbSerialDiscovery discovery;
+
+ private UsbSerialDeviceInformation usb1 = new UsbSerialDeviceInformation(0x100, 0x111, "serial1", "manufacturer1",
+ "product1", 0x1, "interface1", "rfc2217://1.1.1.1:1000");
+ private UsbSerialDeviceInformation usb2 = new UsbSerialDeviceInformation(0x200, 0x222, "serial2", "manufacturer2",
+ "product2", 0x2, "interface2", "rfc2217://[0:0:0:0:0:ffff:0202:0202]:2222");
+ private UsbSerialDeviceInformation usb3 = new UsbSerialDeviceInformation(0x300, 0x333, null, null, null, 0x3, null,
+ "rfc2217://123.222.100.000:3030");
+
+ @BeforeEach
+ public void beforeEach() {
+ discovery = new Ser2NetUsbSerialDiscovery(mdnsClientMock);
+ discovery.registerDiscoveryListener(discoveryListenerMock);
+
+ setupServiceInfo1Mock();
+ setupServiceInfo2Mock();
+ setupServiceInfo3Mock();
+ setupInvalidServiceInfoMock();
+
+ when(serviceEvent1Mock.getInfo()).thenReturn(serviceInfo1Mock);
+ when(serviceEvent2Mock.getInfo()).thenReturn(serviceInfo2Mock);
+ when(serviceEvent3Mock.getInfo()).thenReturn(serviceInfo3Mock);
+ when(invalidServiceEventMock.getInfo()).thenReturn(invalidServiceInfoMock);
+ }
+
+ private void setupServiceInfo1Mock() {
+ when(serviceInfo1Mock.getHostAddresses()).thenReturn(new String[] { "1.1.1.1" });
+ when(serviceInfo1Mock.getPort()).thenReturn(1000);
+
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_VENDOR_ID)).thenReturn("0100");
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_PRODUCT_ID)).thenReturn("0111");
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_SERIAL_NUMBER)).thenReturn("serial1");
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_MANUFACTURER)).thenReturn("manufacturer1");
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_PRODUCT)).thenReturn("product1");
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_INTERFACE_NUMBER)).thenReturn("01");
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_INTERFACE)).thenReturn("interface1");
+
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_PROVIDER)).thenReturn(SER2NET);
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_DEVICE_TYPE)).thenReturn(SERIALUSB);
+ when(serviceInfo1Mock.getPropertyString(PROPERTY_GENSIO_STACK)).thenReturn(TELNET_RFC2217_TCP);
+ }
+
+ private void setupServiceInfo2Mock() {
+ when(serviceInfo2Mock.getHostAddresses()).thenReturn(new String[] { "[0:0:0:0:0:ffff:0202:0202]" });
+ when(serviceInfo2Mock.getPort()).thenReturn(2222);
+
+ when(serviceInfo2Mock.getPropertyString(PROPERTY_VENDOR_ID)).thenReturn("0200");
+ when(serviceInfo2Mock.getPropertyString(PROPERTY_PRODUCT_ID)).thenReturn("0222");
+ when(serviceInfo2Mock.getPropertyString(PROPERTY_SERIAL_NUMBER)).thenReturn("serial2");
+ when(serviceInfo2Mock.getPropertyString(PROPERTY_MANUFACTURER)).thenReturn("manufacturer2");
+ when(serviceInfo2Mock.getPropertyString(PROPERTY_PRODUCT)).thenReturn("product2");
+ when(serviceInfo2Mock.getPropertyString(PROPERTY_INTERFACE_NUMBER)).thenReturn("02");
+ when(serviceInfo2Mock.getPropertyString(PROPERTY_INTERFACE)).thenReturn("interface2");
+ }
+
+ private void setupServiceInfo3Mock() {
+ when(serviceInfo3Mock.getHostAddresses()).thenReturn(new String[] { "123.222.100.000" });
+ when(serviceInfo3Mock.getPort()).thenReturn(3030);
+
+ when(serviceInfo3Mock.getPropertyString(PROPERTY_VENDOR_ID)).thenReturn("0300");
+ when(serviceInfo3Mock.getPropertyString(PROPERTY_PRODUCT_ID)).thenReturn("0333");
+ when(serviceInfo3Mock.getPropertyString(PROPERTY_INTERFACE_NUMBER)).thenReturn("03");
+ }
+
+ private void setupInvalidServiceInfoMock() {
+ when(invalidServiceInfoMock.getHostAddresses()).thenReturn(new String[] { "1.1.1.1" });
+ when(invalidServiceInfoMock.getPort()).thenReturn(1000);
+
+ when(invalidServiceInfoMock.getPropertyString(PROPERTY_VENDOR_ID)).thenReturn("invalid");
+ }
+
+ @Test
+ public void noScansWithoutBackgroundDiscovery() throws InterruptedException {
+ // Wait a little more than one second to give background scanning a chance to kick in.
+ Thread.sleep(1200);
+
+ verify(mdnsClientMock, never()).list(anyString());
+ verify(mdnsClientMock, never()).list(anyString(), ArgumentMatchers.any(Duration.class));
+ }
+
+ @Test
+ public void singleScanReportsResultsCorrectAfterOneScan() {
+ when(mdnsClientMock.list(SERVICE_TYPE, SINGLE_SCAN_DURATION))
+ .thenReturn(new ServiceInfo[] { serviceInfo1Mock, serviceInfo2Mock });
+
+ discovery.doSingleScan();
+
+ // Expectation: discovery listener called with newly discovered devices usb1 and usb2; not called with removed
+ // devices.
+
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb1);
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb2);
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb3);
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(any(UsbSerialDeviceInformation.class));
+ }
+
+ @Test
+ public void singleScanReportsResultsCorrectAfterOneScanWithInvalidServiceInfo() {
+ when(mdnsClientMock.list(SERVICE_TYPE, SINGLE_SCAN_DURATION))
+ .thenReturn(new ServiceInfo[] { serviceInfo1Mock, invalidServiceInfoMock, serviceInfo2Mock });
+
+ discovery.doSingleScan();
+
+ // Expectation: discovery listener is still called with newly discovered devices usb1 and usb2; not called with
+ // removed devices.
+
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb1);
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb2);
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb3);
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(any(UsbSerialDeviceInformation.class));
+ }
+
+ @Test
+ public void singleScanReportsResultsCorrectlyAfterTwoScans() {
+ when(mdnsClientMock.list(SERVICE_TYPE, SINGLE_SCAN_DURATION))
+ .thenReturn(new ServiceInfo[] { serviceInfo1Mock, serviceInfo2Mock })
+ .thenReturn(new ServiceInfo[] { serviceInfo2Mock, serviceInfo3Mock });
+
+ discovery.unregisterDiscoveryListener(discoveryListenerMock);
+ discovery.doSingleScan();
+
+ discovery.registerDiscoveryListener(discoveryListenerMock);
+ discovery.doSingleScan();
+
+ // Expectation: discovery listener called once for removing usb1, and once for adding usb2/usb3 each.
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb1);
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceRemoved(usb1);
+
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb2);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb2);
+
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb3);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb3);
+ }
+
+ @Test
+ public void backgroundScanning() {
+ discovery.startBackgroundScanning();
+
+ discovery.serviceAdded(serviceEvent1Mock);
+ discovery.serviceRemoved(serviceEvent1Mock);
+ discovery.serviceAdded(serviceEvent2Mock);
+ discovery.serviceAdded(invalidServiceEventMock);
+ discovery.serviceResolved(serviceEvent3Mock);
+
+ discovery.stopBackgroundScanning();
+
+ // Expectation: discovery listener called once for each discovered device, and once for removal of usb1.
+
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb1);
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceRemoved(usb1);
+
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb2);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb2);
+
+ verify(discoveryListenerMock, times(1)).usbSerialDeviceDiscovered(usb3);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb3);
+ }
+
+ @Test
+ public void noBackgroundScanning() throws IOException, InterruptedException {
+ discovery.stopBackgroundScanning();
+
+ discovery.serviceAdded(serviceEvent1Mock);
+ discovery.serviceRemoved(serviceEvent1Mock);
+ discovery.serviceAdded(serviceEvent2Mock);
+ discovery.serviceResolved(serviceEvent3Mock);
+
+ // Expectation: discovery listener is never called when there is no background scanning is.
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb1);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb1);
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb2);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb2);
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb3);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb3);
+ }
+
+ @Test
+ public void discoveryChecksSer2NetSpecificProperties() {
+ discovery.startBackgroundScanning();
+
+ when(serviceInfo3Mock.getPropertyString(PROPERTY_PROVIDER)).thenReturn(SER2NET);
+ when(serviceInfo3Mock.getPropertyString(PROPERTY_GENSIO_STACK)).thenReturn("incompatible");
+
+ discovery.serviceAdded(serviceEvent3Mock);
+
+ when(serviceInfo3Mock.getPropertyString(PROPERTY_PROVIDER)).thenReturn(SER2NET);
+ when(serviceInfo3Mock.getPropertyString(PROPERTY_DEVICE_TYPE)).thenReturn("incompatible");
+
+ discovery.serviceAdded(serviceEvent3Mock);
+
+ // Expectation: discovery listener is never called when the ser2net specific properties do not match.
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb1);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb1);
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb2);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb2);
+
+ verify(discoveryListenerMock, never()).usbSerialDeviceDiscovered(usb3);
+ verify(discoveryListenerMock, never()).usbSerialDeviceRemoved(usb3);
+ }
+}
diff --git a/bundles/org.openhab.core.config.discovery.usbserial/src/main/java/org/openhab/core/config/discovery/usbserial/internal/UsbSerialDiscoveryService.java b/bundles/org.openhab.core.config.discovery.usbserial/src/main/java/org/openhab/core/config/discovery/usbserial/internal/UsbSerialDiscoveryService.java
index 69893fbf9c0..676e64a9e2a 100644
--- a/bundles/org.openhab.core.config.discovery.usbserial/src/main/java/org/openhab/core/config/discovery/usbserial/internal/UsbSerialDiscoveryService.java
+++ b/bundles/org.openhab.core.config.discovery.usbserial/src/main/java/org/openhab/core/config/discovery/usbserial/internal/UsbSerialDiscoveryService.java
@@ -47,14 +47,14 @@
* This discovery service is intended to be used by bindings that support USB devices, but do not directly talk to the
* USB devices but rather use a serial port for the communication, where the serial port is provided by an operating
* system driver outside the scope of openHAB. Examples for such USB devices are USB dongles that provide
- * access to wireless networks, like, e.g., Zigbeee or Zwave dongles.
+ * access to wireless networks, like, e.g., Zigbee or Zwave dongles.
*
* This discovery service provides functionality for discovering added and removed USB devices and the corresponding
* serial ports. The actual {@link DiscoveryResult}s are then provided by {@link UsbSerialDiscoveryParticipant}s, which
* are called by this discovery service whenever new devices are detected or devices are removed. Such
* {@link UsbSerialDiscoveryParticipant}s should be provided by bindings accessing USB devices via a serial port.
*
- * This discovery service requires a component implementing the interface {@link UsbSerialDiscovery}, which performs the
+ * This discovery service requires components implementing the interface {@link UsbSerialDiscovery}, which perform the
* actual serial port and USB device discovery (as this discovery might differ depending on the operating system).
*
* @author Henning Sudbrock - Initial contribution
@@ -70,10 +70,8 @@ public class UsbSerialDiscoveryService extends AbstractDiscoveryService implemen
private static final String THING_PROPERTY_USB_PRODUCT_ID = "usb_product_id";
private final Set discoveryParticipants = new CopyOnWriteArraySet<>();
-
private final Set previouslyDiscovered = new CopyOnWriteArraySet<>();
-
- private @NonNullByDefault({}) UsbSerialDiscovery usbSerialDiscovery;
+ private final Set usbSerialDiscoveries = new CopyOnWriteArraySet<>();
public UsbSerialDiscoveryService() {
super(5);
@@ -83,7 +81,6 @@ public UsbSerialDiscoveryService() {
@Activate
protected void activate(@Nullable Map configProperties) {
super.activate(configProperties);
- usbSerialDiscovery.registerDiscoveryListener(this);
}
@Modified
@@ -100,7 +97,7 @@ protected void deactivate() {
@Reference(cardinality = ReferenceCardinality.MULTIPLE, policy = ReferencePolicy.DYNAMIC)
protected void addUsbSerialDiscoveryParticipant(UsbSerialDiscoveryParticipant participant) {
- this.discoveryParticipants.add(participant);
+ discoveryParticipants.add(participant);
for (UsbSerialDeviceInformation usbSerialDeviceInformation : previouslyDiscovered) {
DiscoveryResult result = participant.createResult(usbSerialDeviceInformation);
if (result != null) {
@@ -110,19 +107,23 @@ protected void addUsbSerialDiscoveryParticipant(UsbSerialDiscoveryParticipant pa
}
protected void removeUsbSerialDiscoveryParticipant(UsbSerialDiscoveryParticipant participant) {
- this.discoveryParticipants.remove(participant);
+ discoveryParticipants.remove(participant);
}
- @Reference
- protected void setUsbSerialDiscovery(UsbSerialDiscovery usbSerialDiscovery) {
- this.usbSerialDiscovery = usbSerialDiscovery;
+ @Reference(cardinality = ReferenceCardinality.MULTIPLE, policy = ReferencePolicy.DYNAMIC)
+ protected void addUsbSerialDiscovery(UsbSerialDiscovery usbSerialDiscovery) {
+ usbSerialDiscoveries.add(usbSerialDiscovery);
+ usbSerialDiscovery.registerDiscoveryListener(this);
+ if (isBackgroundDiscoveryEnabled()) {
+ usbSerialDiscovery.startBackgroundScanning();
+ }
}
- protected synchronized void unsetUsbSerialDiscovery(UsbSerialDiscovery usbSerialDiscovery) {
+ protected synchronized void removeUsbSerialDiscovery(UsbSerialDiscovery usbSerialDiscovery) {
usbSerialDiscovery.stopBackgroundScanning();
usbSerialDiscovery.unregisterDiscoveryListener(this);
- this.usbSerialDiscovery = null;
- this.previouslyDiscovered.clear();
+ usbSerialDiscoveries.remove(usbSerialDiscovery);
+ previouslyDiscovered.clear();
}
@Override
@@ -133,30 +134,17 @@ public Set getSupportedThingTypes() {
@Override
protected void startScan() {
- if (usbSerialDiscovery != null) {
- usbSerialDiscovery.doSingleScan();
- } else {
- logger.info("Could not scan, as there is no USB-Serial discovery service configured.");
- }
+ usbSerialDiscoveries.forEach(UsbSerialDiscovery::doSingleScan);
}
@Override
protected void startBackgroundDiscovery() {
- if (usbSerialDiscovery != null) {
- usbSerialDiscovery.startBackgroundScanning();
- } else {
- logger.info(
- "Could not start background discovery, as there is no USB-Serial discovery service configured.");
- }
+ usbSerialDiscoveries.forEach(UsbSerialDiscovery::startBackgroundScanning);
}
@Override
protected void stopBackgroundDiscovery() {
- if (usbSerialDiscovery != null) {
- usbSerialDiscovery.stopBackgroundScanning();
- } else {
- logger.info("Could not stop background discovery, as there is no USB-Serial discovery service configured.");
- }
+ usbSerialDiscoveries.forEach(UsbSerialDiscovery::stopBackgroundScanning);
}
@Override
diff --git a/bundles/org.openhab.core.config.serial/pom.xml b/bundles/org.openhab.core.config.serial/pom.xml
index 83f4ea7380c..22e833f0ed5 100644
--- a/bundles/org.openhab.core.config.serial/pom.xml
+++ b/bundles/org.openhab.core.config.serial/pom.xml
@@ -25,6 +25,11 @@
org.openhab.core.io.transport.serial
${project.version}
+
+ org.openhab.core.bundles
+ org.openhab.core.config.discovery.usbserial
+ ${project.version}
+
diff --git a/bundles/org.openhab.core.config.serial/src/main/java/org/openhab/core/config/serial/internal/SerialConfigOptionProvider.java b/bundles/org.openhab.core.config.serial/src/main/java/org/openhab/core/config/serial/internal/SerialConfigOptionProvider.java
index 99411929586..2e0449bf57c 100644
--- a/bundles/org.openhab.core.config.serial/src/main/java/org/openhab/core/config/serial/internal/SerialConfigOptionProvider.java
+++ b/bundles/org.openhab.core.config.serial/src/main/java/org/openhab/core/config/serial/internal/SerialConfigOptionProvider.java
@@ -13,43 +13,82 @@
package org.openhab.core.config.serial.internal;
import java.net.URI;
-import java.util.ArrayList;
import java.util.Collection;
-import java.util.List;
import java.util.Locale;
+import java.util.Set;
+import java.util.concurrent.CopyOnWriteArraySet;
+import java.util.stream.Collectors;
+import java.util.stream.Stream;
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.eclipse.jdt.annotation.Nullable;
import org.openhab.core.config.core.ConfigOptionProvider;
import org.openhab.core.config.core.ParameterOption;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDeviceInformation;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDiscovery;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDiscoveryListener;
+import org.openhab.core.io.transport.serial.SerialPortIdentifier;
import org.openhab.core.io.transport.serial.SerialPortManager;
+import org.osgi.service.component.annotations.Activate;
import org.osgi.service.component.annotations.Component;
import org.osgi.service.component.annotations.Reference;
+import org.osgi.service.component.annotations.ReferenceCardinality;
+import org.osgi.service.component.annotations.ReferencePolicy;
/**
* This service provides serial port names as options for configuration parameters.
*
* @author Kai Kreuzer - Initial contribution
+ * @author Wouter Born - Add discovered USB serial port names to serial port parameter options
*/
+@NonNullByDefault
@Component
-public class SerialConfigOptionProvider implements ConfigOptionProvider {
+public class SerialConfigOptionProvider implements ConfigOptionProvider, UsbSerialDiscoveryListener {
- private SerialPortManager serialPortManager;
+ static final String SERIAL_PORT = "serial-port";
- @Reference
- protected void setSerialPortManager(final SerialPortManager serialPortManager) {
+ private final SerialPortManager serialPortManager;
+ private final Set previouslyDiscovered = new CopyOnWriteArraySet<>();
+ private final Set usbSerialDiscoveries = new CopyOnWriteArraySet<>();
+
+ @Activate
+ public SerialConfigOptionProvider(final @Reference SerialPortManager serialPortManager) {
this.serialPortManager = serialPortManager;
}
- protected void unsetSerialPortManager(final SerialPortManager serialPortManager) {
- this.serialPortManager = null;
+ @Reference(cardinality = ReferenceCardinality.MULTIPLE, policy = ReferencePolicy.DYNAMIC)
+ protected void addUsbSerialDiscovery(UsbSerialDiscovery usbSerialDiscovery) {
+ usbSerialDiscoveries.add(usbSerialDiscovery);
+ usbSerialDiscovery.registerDiscoveryListener(this);
+ }
+
+ protected synchronized void removeUsbSerialDiscovery(UsbSerialDiscovery usbSerialDiscovery) {
+ usbSerialDiscovery.unregisterDiscoveryListener(this);
+ usbSerialDiscoveries.remove(usbSerialDiscovery);
+ previouslyDiscovered.clear();
+ }
+
+ @Override
+ public void usbSerialDeviceDiscovered(UsbSerialDeviceInformation usbSerialDeviceInformation) {
+ previouslyDiscovered.add(usbSerialDeviceInformation);
+ }
+
+ @Override
+ public void usbSerialDeviceRemoved(UsbSerialDeviceInformation usbSerialDeviceInformation) {
+ previouslyDiscovered.remove(usbSerialDeviceInformation);
}
@Override
- public Collection getParameterOptions(URI uri, String param, String context, Locale locale) {
- List options = new ArrayList<>();
- if ("serial-port".equals(context)) {
- serialPortManager.getIdentifiers()
- .forEach(id -> options.add(new ParameterOption(id.getName(), id.getName())));
+ public @Nullable Collection getParameterOptions(URI uri, String param, @Nullable String context,
+ @Nullable Locale locale) {
+ if (SERIAL_PORT.equals(context)) {
+ return Stream
+ .concat(serialPortManager.getIdentifiers().map(SerialPortIdentifier::getName),
+ previouslyDiscovered.stream().map(UsbSerialDeviceInformation::getSerialPort))
+ .distinct() //
+ .map(serialPortName -> new ParameterOption(serialPortName, serialPortName)) //
+ .collect(Collectors.toList());
}
- return options;
+ return null;
}
}
diff --git a/bundles/org.openhab.core.config.serial/src/test/java/org/openhab/core/config/serial/internal/SerialConfigOptionProviderTest.java b/bundles/org.openhab.core.config.serial/src/test/java/org/openhab/core/config/serial/internal/SerialConfigOptionProviderTest.java
new file mode 100644
index 00000000000..c5394d36e67
--- /dev/null
+++ b/bundles/org.openhab.core.config.serial/src/test/java/org/openhab/core/config/serial/internal/SerialConfigOptionProviderTest.java
@@ -0,0 +1,178 @@
+/**
+ * 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.core.config.serial.internal;
+
+import static org.hamcrest.CoreMatchers.*;
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.mockito.Mockito.when;
+import static org.openhab.core.config.serial.internal.SerialConfigOptionProvider.SERIAL_PORT;
+
+import java.net.URI;
+import java.util.Arrays;
+import java.util.Collection;
+import java.util.stream.Collectors;
+import java.util.stream.Stream;
+
+import org.eclipse.jdt.annotation.NonNullByDefault;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.Mock;
+import org.mockito.junit.jupiter.MockitoExtension;
+import org.mockito.junit.jupiter.MockitoSettings;
+import org.mockito.quality.Strictness;
+import org.openhab.core.config.core.ParameterOption;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDeviceInformation;
+import org.openhab.core.config.discovery.usbserial.UsbSerialDiscovery;
+import org.openhab.core.io.transport.serial.SerialPortIdentifier;
+import org.openhab.core.io.transport.serial.SerialPortManager;
+
+/**
+ * Unit tests for the {@link SerialConfigOptionProvider}.
+ *
+ * @author Wouter Born - Initial contribution
+ */
+@NonNullByDefault
+@ExtendWith(MockitoExtension.class)
+@MockitoSettings(strictness = Strictness.LENIENT)
+public class SerialConfigOptionProviderTest {
+
+ private static final String DEV_TTY_S1 = "/dev/ttyS1";
+ private static final String DEV_TTY_S2 = "/dev/ttyS2";
+ private static final String DEV_TTY_S3 = "/dev/ttyS3";
+
+ private static final String RFC2217_IPV4 = "rfc2217://1.1.1.1:1000";
+ private static final String RFC2217_IPV6 = "rfc2217://[0:0:0:0:0:ffff:0202:0202]:2222";
+
+ private UsbSerialDeviceInformation usb1 = new UsbSerialDeviceInformation(0x100, 0x111, "serial1", "manufacturer1",
+ "product1", 0x1, "interface1", RFC2217_IPV4);
+ private UsbSerialDeviceInformation usb2 = new UsbSerialDeviceInformation(0x200, 0x222, "serial2", "manufacturer2",
+ "product2", 0x2, "interface2", RFC2217_IPV6);
+ private UsbSerialDeviceInformation usb3 = new UsbSerialDeviceInformation(0x300, 0x333, "serial3", "manufacturer3",
+ "product3", 0x3, "interface3", DEV_TTY_S3);
+
+ private @Mock @NonNullByDefault({}) SerialPortManager serialPortManagerMock;
+ private @Mock @NonNullByDefault({}) UsbSerialDiscovery usbSerialDiscoveryMock;
+
+ private @Mock @NonNullByDefault({}) SerialPortIdentifier serialPortIdentifier1Mock;
+ private @Mock @NonNullByDefault({}) SerialPortIdentifier serialPortIdentifier2Mock;
+ private @Mock @NonNullByDefault({}) SerialPortIdentifier serialPortIdentifier3Mock;
+
+ private @NonNullByDefault({}) SerialConfigOptionProvider provider;
+
+ @BeforeEach
+ public void beforeEach() {
+ provider = new SerialConfigOptionProvider(serialPortManagerMock);
+
+ when(serialPortIdentifier1Mock.getName()).thenReturn(DEV_TTY_S1);
+ when(serialPortIdentifier2Mock.getName()).thenReturn(DEV_TTY_S2);
+ when(serialPortIdentifier3Mock.getName()).thenReturn(DEV_TTY_S3);
+ }
+
+ private void assertParameterOptions(String... serialPortIdentifiers) {
+ Collection actual = provider.getParameterOptions(URI.create("uri"), "serialPort", SERIAL_PORT,
+ null);
+ Collection expected = Arrays.stream(serialPortIdentifiers)
+ .map(id -> new ParameterOption(id, id)).collect(Collectors.toList());
+ assertThat(actual, is(expected));
+ }
+
+ @Test
+ public void noSerialPortIdentifiers() {
+ when(serialPortManagerMock.getIdentifiers()).thenReturn(Stream.of());
+ assertParameterOptions();
+ }
+
+ @Test
+ public void serialPortManagerIdentifiersOnly() {
+ when(serialPortManagerMock.getIdentifiers())
+ .thenReturn(Stream.of(serialPortIdentifier1Mock, serialPortIdentifier2Mock));
+
+ assertParameterOptions(DEV_TTY_S1, DEV_TTY_S2);
+ }
+
+ @Test
+ public void discoveredIdentifiersOnly() {
+ provider.addUsbSerialDiscovery(usbSerialDiscoveryMock);
+
+ provider.usbSerialDeviceDiscovered(usb1);
+ provider.usbSerialDeviceDiscovered(usb2);
+
+ when(serialPortManagerMock.getIdentifiers()).thenReturn(Stream.of());
+
+ assertParameterOptions(RFC2217_IPV4, RFC2217_IPV6);
+ }
+
+ @Test
+ public void serialPortManagerAndDiscoveredIdentifiers() {
+ provider.addUsbSerialDiscovery(usbSerialDiscoveryMock);
+
+ provider.usbSerialDeviceDiscovered(usb1);
+ provider.usbSerialDeviceDiscovered(usb2);
+
+ when(serialPortManagerMock.getIdentifiers())
+ .thenReturn(Stream.of(serialPortIdentifier1Mock, serialPortIdentifier2Mock));
+
+ assertParameterOptions(DEV_TTY_S1, DEV_TTY_S2, RFC2217_IPV4, RFC2217_IPV6);
+ }
+
+ @Test
+ public void removedDevicesAreRemoved() {
+ provider.addUsbSerialDiscovery(usbSerialDiscoveryMock);
+
+ provider.usbSerialDeviceDiscovered(usb1);
+
+ when(serialPortManagerMock.getIdentifiers()).thenReturn(Stream.of());
+ assertParameterOptions(RFC2217_IPV4);
+
+ provider.usbSerialDeviceRemoved(usb1);
+
+ when(serialPortManagerMock.getIdentifiers()).thenReturn(Stream.of());
+ assertParameterOptions();
+ }
+
+ @Test
+ public void discoveryRemovalClearsDiscoveryResults() {
+ provider.addUsbSerialDiscovery(usbSerialDiscoveryMock);
+
+ provider.usbSerialDeviceDiscovered(usb1);
+ provider.usbSerialDeviceDiscovered(usb2);
+ provider.usbSerialDeviceDiscovered(usb3);
+
+ when(serialPortManagerMock.getIdentifiers()).thenReturn(Stream.of());
+ assertParameterOptions(RFC2217_IPV4, RFC2217_IPV6, DEV_TTY_S3);
+
+ provider.removeUsbSerialDiscovery(usbSerialDiscoveryMock);
+
+ when(serialPortManagerMock.getIdentifiers()).thenReturn(Stream.of());
+ assertParameterOptions();
+ }
+
+ @Test
+ public void serialPortIdentifiersAreUnique() {
+ provider.addUsbSerialDiscovery(usbSerialDiscoveryMock);
+
+ provider.usbSerialDeviceDiscovered(usb3);
+
+ when(serialPortManagerMock.getIdentifiers()).thenReturn(Stream.of(serialPortIdentifier3Mock));
+
+ assertParameterOptions(DEV_TTY_S3);
+ }
+
+ @Test
+ public void nullResultIfContextDoesNotMatch() {
+ Collection actual = provider.getParameterOptions(URI.create("uri"), "serialPort",
+ "otherContext", null);
+ assertThat(actual, is(nullValue()));
+ }
+}
diff --git a/bundles/org.openhab.core.io.transport.serial.rxtx.rfc2217/src/main/java/org/openhab/core/io/transport/serial/rxtx/rfc2217/internal/RFC2217PortProvider.java b/bundles/org.openhab.core.io.transport.serial.rxtx.rfc2217/src/main/java/org/openhab/core/io/transport/serial/rxtx/rfc2217/internal/RFC2217PortProvider.java
index fef063d36a3..1c25fb9ace6 100644
--- a/bundles/org.openhab.core.io.transport.serial.rxtx.rfc2217/src/main/java/org/openhab/core/io/transport/serial/rxtx/rfc2217/internal/RFC2217PortProvider.java
+++ b/bundles/org.openhab.core.io.transport.serial.rxtx.rfc2217/src/main/java/org/openhab/core/io/transport/serial/rxtx/rfc2217/internal/RFC2217PortProvider.java
@@ -49,7 +49,6 @@ public Stream getAcceptedProtocols() {
@Override
public Stream getSerialPortIdentifiers() {
- // TODO implement discovery here. /~https://github.com/eclipse/smarthome/pull/5560
return Stream.empty();
}
}
diff --git a/bundles/pom.xml b/bundles/pom.xml
index 101aeca02aa..748a9326d26 100644
--- a/bundles/pom.xml
+++ b/bundles/pom.xml
@@ -31,6 +31,7 @@
org.openhab.core.config.discovery.mdns
org.openhab.core.config.discovery.usbserial
org.openhab.core.config.discovery.usbserial.linuxsysfs
+ org.openhab.core.config.discovery.usbserial.ser2net
org.openhab.core.config.discovery.upnp
org.openhab.core.config.dispatch
org.openhab.core.config.serial
diff --git a/features/karaf/openhab-core/src/main/feature/feature.xml b/features/karaf/openhab-core/src/main/feature/feature.xml
index c6af467b5b1..bf21e7f26bf 100644
--- a/features/karaf/openhab-core/src/main/feature/feature.xml
+++ b/features/karaf/openhab-core/src/main/feature/feature.xml
@@ -463,9 +463,12 @@
openhab.tp;filter:="(&(feature=serial)(impl=rxtx))"
openhab.tp-serial-rxtx
+ openhab-core-io-transport-mdns
+
mvn:org.openhab.core.bundles/org.openhab.core.config.serial/${project.version}
mvn:org.openhab.core.bundles/org.openhab.core.config.discovery.usbserial/${project.version}
mvn:org.openhab.core.bundles/org.openhab.core.config.discovery.usbserial.linuxsysfs/${project.version}
+ mvn:org.openhab.core.bundles/org.openhab.core.config.discovery.usbserial.ser2net/${project.version}
mvn:org.openhab.core.bundles/org.openhab.core.io.transport.serial/${project.version}
mvn:org.openhab.core.bundles/org.openhab.core.io.transport.serial.rxtx/${project.version}
mvn:org.openhab.core.bundles/org.openhab.core.io.transport.serial.rxtx.rfc2217/${project.version}