2 * Copyright (c) 2010-2022 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.epsonprojector.internal.discovery;
15 import static org.openhab.binding.epsonprojector.internal.EpsonProjectorBindingConstants.*;
17 import java.io.IOException;
18 import java.net.SocketException;
21 import java.util.concurrent.ScheduledFuture;
22 import java.util.concurrent.TimeUnit;
24 import org.eclipse.jdt.annotation.NonNullByDefault;
25 import org.eclipse.jdt.annotation.Nullable;
26 import org.openhab.core.config.discovery.AbstractDiscoveryService;
27 import org.openhab.core.config.discovery.DiscoveryResultBuilder;
28 import org.openhab.core.config.discovery.DiscoveryService;
29 import org.openhab.core.net.NetworkAddressService;
30 import org.openhab.core.thing.ThingTypeUID;
31 import org.openhab.core.thing.ThingUID;
32 import org.osgi.service.component.annotations.Activate;
33 import org.osgi.service.component.annotations.Component;
34 import org.osgi.service.component.annotations.Reference;
35 import org.slf4j.Logger;
36 import org.slf4j.LoggerFactory;
39 * The {@link EpsonProjectoreDiscoveryService} class implements a service
40 * for discovering Epson projectors using the AMX Device Discovery protocol.
42 * @author Mark Hilbush - Initial contribution
43 * @author Michael Lobstein - Adapted for the Epson Projector binding
46 @Component(service = DiscoveryService.class, configurationPid = "discovery.epsonprojector")
47 public class EpsonProjectorDiscoveryService extends AbstractDiscoveryService {
48 private final Logger logger = LoggerFactory.getLogger(EpsonProjectorDiscoveryService.class);
49 private @Nullable ScheduledFuture<?> epsonDiscoveryJob;
51 // Discovery parameters
52 public static final boolean BACKGROUND_DISCOVERY_ENABLED = true;
53 public static final int BACKGROUND_DISCOVERY_DELAY_TIMEOUT_SEC = 10;
55 private NetworkAddressService networkAddressService;
57 private boolean terminate = false;
60 public EpsonProjectorDiscoveryService(@Reference NetworkAddressService networkAddressService) {
61 super(SUPPORTED_THING_TYPES_UIDS, 0, BACKGROUND_DISCOVERY_ENABLED);
62 this.networkAddressService = networkAddressService;
63 epsonDiscoveryJob = null;
68 public Set<ThingTypeUID> getSupportedThingTypes() {
69 return SUPPORTED_THING_TYPES_UIDS;
73 protected void startBackgroundDiscovery() {
74 if (epsonDiscoveryJob == null) {
76 logger.debug("Starting background discovery job in {} seconds", BACKGROUND_DISCOVERY_DELAY_TIMEOUT_SEC);
77 epsonDiscoveryJob = scheduler.schedule(this::discover, BACKGROUND_DISCOVERY_DELAY_TIMEOUT_SEC,
83 protected void stopBackgroundDiscovery() {
84 ScheduledFuture<?> epsonDiscoveryJob = this.epsonDiscoveryJob;
85 if (epsonDiscoveryJob != null) {
87 epsonDiscoveryJob.cancel(false);
88 this.epsonDiscoveryJob = null;
93 public void startScan() {
97 public void stopScan() {
100 private synchronized void discover() {
101 logger.debug("Discovery job is running");
102 MulticastListener epsonMulticastListener;
103 String local = "127.0.0.1";
106 String ip = networkAddressService.getPrimaryIpv4HostAddress();
107 epsonMulticastListener = new MulticastListener((ip != null ? ip : local));
108 } catch (SocketException se) {
109 logger.debug("Discovery job got Socket exception creating multicast socket: {}", se.getMessage());
111 } catch (IOException ioe) {
112 logger.debug("Discovery job got IO exception creating multicast socket: {}", ioe.getMessage());
118 // Wait for a discovery beacon to return properties for an Epson projector.
119 Map<String, Object> thingProperties = epsonMulticastListener.waitForBeacon();
121 if (thingProperties != null) {
122 // The MulticastListener found a projector, add it as new thing
123 String uid = (String) thingProperties.get(THING_PROPERTY_MAC);
124 String ipAddress = (String) thingProperties.get(THING_PROPERTY_HOST);
127 logger.trace("Projector with UID {} discovered at IP: {}", uid, ipAddress);
129 ThingUID thingUid = new ThingUID(THING_TYPE_PROJECTOR_TCP, uid);
130 logger.trace("Creating epson projector discovery result for: {}, IP={}", uid, ipAddress);
131 thingDiscovered(DiscoveryResultBuilder.create(thingUid).withProperties(thingProperties)
132 .withLabel("Epson Projector " + uid).withRepresentationProperty(THING_PROPERTY_MAC)
136 } catch (IOException ioe) {
137 logger.debug("Discovery job got exception waiting for beacon: {}", ioe.getMessage());
140 epsonMulticastListener.shutdown();
141 logger.debug("Discovery job is exiting");