[tplinkrouter] Initial contribution (#13369)
* Initial commit Signed-off-by: Olivier Marceau <hollysaiqs@marceau.ovh>pull/12803/head
parent
19b4c95103
commit
73e18424b9
|
@ -329,6 +329,7 @@
|
|||
/bundles/org.openhab.binding.tibber/ @kjoglum
|
||||
/bundles/org.openhab.binding.tivo/ @mlobstein
|
||||
/bundles/org.openhab.binding.touchwand/ @roieg
|
||||
/bundles/org.openhab.binding.tplinkrouter/ @olivierkeke
|
||||
/bundles/org.openhab.binding.tplinksmarthome/ @Hilbrand
|
||||
/bundles/org.openhab.binding.tr064/ @openhab/add-ons-maintainers
|
||||
/bundles/org.openhab.binding.tradfri/ @cweitkamp @kaikreuzer
|
||||
|
|
|
@ -1651,6 +1651,11 @@
|
|||
<artifactId>org.openhab.binding.touchwand</artifactId>
|
||||
<version>${project.version}</version>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.openhab.addons.bundles</groupId>
|
||||
<artifactId>org.openhab.binding.tplinkrouter</artifactId>
|
||||
<version>${project.version}</version>
|
||||
</dependency>
|
||||
<dependency>
|
||||
<groupId>org.openhab.addons.bundles</groupId>
|
||||
<artifactId>org.openhab.binding.tplinksmarthome</artifactId>
|
||||
|
|
|
@ -0,0 +1,13 @@
|
|||
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-addons
|
|
@ -0,0 +1,57 @@
|
|||
# tplinkrouter Binding
|
||||
|
||||
The tplinkrouter Binding allows monitoring and controlling TP-Link routers.
|
||||
|
||||
The binding uses a telnet connection to communicate with the router.
|
||||
|
||||
At the moment only wifi part is supported and `TD-W9970` is the only model tested.
|
||||
This binding may work with other TP-Link router provided that they use the same telnet API.
|
||||
|
||||
## Supported Things
|
||||
|
||||
This binding provides only the `router` Thing.
|
||||
|
||||
## Thing Configuration
|
||||
|
||||
### `router` Thing Configuration
|
||||
|
||||
| Name | Type | Description | Default | Required | Advanced |
|
||||
|-----------------|---------|-----------------------------------------------|---------|----------|----------|
|
||||
| hostname | text | Hostname or IP address of the device | N/A | yes | no |
|
||||
| port | integer | Port for telnet connection | 23 | no | no |
|
||||
| username | text | Username to access the router (same as WebUI) | N/A | yes | no |
|
||||
| password | text | Password to access the device (same as WebUI) | N/A | yes | no |
|
||||
| refreshInterval | integer | Interval the device is polled in sec. | 60 | no | yes |
|
||||
|
||||
## Channels
|
||||
|
||||
| Channel | Type | Read/Write | Description |
|
||||
|-----------------------|--------|------------|------------------------------------------|
|
||||
| `wifi#status` | Switch | RW | State of the wifi |
|
||||
| `wifi#ssid` | String | R | SSID of the wifi network |
|
||||
| `wifi#bandwidth` | String | R | Bandwidth of the wifi network |
|
||||
| `wifi#qss` | Switch | RW | Quick Security Setup of the wifi network |
|
||||
| `wifi#secMode` | String | R | Security Mode of the wifi network |
|
||||
| `wifi#authentication` | String | R | Authentication Mode of the wifi network |
|
||||
| `wifi#encryption` | String | R | Encryption Mode of the wifi network |
|
||||
| `wifi#key` | String | R | Password of the wifi network |
|
||||
|
||||
## Full Example
|
||||
|
||||
`.things` configuration file:
|
||||
|
||||
```
|
||||
Thing tplinkrouter:router:myRouter [hostname="192.168.0.1", username="admin", password="myPassword"]
|
||||
```
|
||||
|
||||
`.items` configuration file:
|
||||
|
||||
```
|
||||
Switch Wifi "Wifi" <QualityOfService> {channel="tplinkrouter:router:myRouter:wifi#status", autoupdate="false"}
|
||||
String WifiSSID "Wifi SSID" <QualityOfService> {channel="tplinkrouter:router:myRouter:wifi#ssid"}
|
||||
String BandWidth "Wifi Bandwidth" <QualityOfService> {channel="tplinkrouter:router:myRouter:wifi#bandwidth"}
|
||||
Switch QSS "Wifi QSS" <QualityOfService> {channel="tplinkrouter:router:myRouter:wifi#qss", autoupdate="false"}
|
||||
String SecMode "Wifi Security Mode" <QualityOfService> {channel="tplinkrouter:router:myRouter:wifi#secMode"}
|
||||
String Authentication "Wifi Authentication Mode" <QualityOfService> {channel="tplinkrouter:router:myRouter:wifi#authentication"}
|
||||
String Encryption "Wifi Encryption Mode" <QualityOfService> {channel="tplinkrouter:router:myRouter:wifi#encryption"}
|
||||
```
|
|
@ -0,0 +1,17 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://maven.apache.org/POM/4.0.0"
|
||||
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
|
||||
|
||||
<modelVersion>4.0.0</modelVersion>
|
||||
|
||||
<parent>
|
||||
<groupId>org.openhab.addons.bundles</groupId>
|
||||
<artifactId>org.openhab.addons.reactor.bundles</artifactId>
|
||||
<version>3.4.0-SNAPSHOT</version>
|
||||
</parent>
|
||||
|
||||
<artifactId>org.openhab.binding.tplinkrouter</artifactId>
|
||||
|
||||
<name>openHAB Add-ons :: Bundles :: TpLinkRouter Binding</name>
|
||||
|
||||
</project>
|
|
@ -0,0 +1,9 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<features name="org.openhab.binding.tplinkrouter-${project.version}" xmlns="http://karaf.apache.org/xmlns/features/v1.4.0">
|
||||
<repository>mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features</repository>
|
||||
|
||||
<feature name="openhab-binding-tplinkrouter" description="TpLinkRouter Binding" version="${project.version}">
|
||||
<feature>openhab-runtime-base</feature>
|
||||
<bundle start-level="80">mvn:org.openhab.addons.bundles/org.openhab.binding.tplinkrouter/${project.version}</bundle>
|
||||
</feature>
|
||||
</features>
|
|
@ -0,0 +1,41 @@
|
|||
/**
|
||||
* Copyright (c) 2010-2022 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.binding.tplinkrouter.internal;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.openhab.core.thing.ThingTypeUID;
|
||||
|
||||
/**
|
||||
* The {@link TpLinkRouterBindingConstants} class defines common constants, which are
|
||||
* used across the whole binding.
|
||||
*
|
||||
* @author Olivier Marceau - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class TpLinkRouterBindingConstants {
|
||||
|
||||
private static final String BINDING_ID = "tplinkrouter";
|
||||
|
||||
// List of all Thing Type UIDs
|
||||
public static final ThingTypeUID THING_TYPE_ROUTER = new ThingTypeUID(BINDING_ID, "router");
|
||||
|
||||
// List of all Channel ids
|
||||
public static final String WIFI_STATUS = "wifi#status";
|
||||
public static final String WIFI_SSID = "wifi#ssid";
|
||||
public static final String WIFI_BANDWIDTH = "wifi#bandwidth";
|
||||
public static final String WIFI_QSS = "wifi#qss";
|
||||
public static final String WIFI_SECMODE = "wifi#secMode";
|
||||
public static final String WIFI_AUTHENTICATION = "wifi#authentication";
|
||||
public static final String WIFI_ENCRYPTION = "wifi#encryption";
|
||||
public static final String WIFI_KEY = "wifi#key";
|
||||
}
|
|
@ -0,0 +1,30 @@
|
|||
/**
|
||||
* Copyright (c) 2010-2022 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.binding.tplinkrouter.internal;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The {@link TpLinkRouterConfiguration} class contains fields mapping thing configuration parameters.
|
||||
*
|
||||
* @author Olivier Marceau - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class TpLinkRouterConfiguration {
|
||||
|
||||
public String hostname = "";
|
||||
public int port = 23;
|
||||
public String username = "";
|
||||
public String password = "";
|
||||
public int refreshInterval = 60;
|
||||
}
|
|
@ -0,0 +1,239 @@
|
|||
/**
|
||||
* Copyright (c) 2010-2022 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.binding.tplinkrouter.internal;
|
||||
|
||||
import static org.openhab.binding.tplinkrouter.internal.TpLinkRouterBindingConstants.*;
|
||||
|
||||
import java.io.IOException;
|
||||
import java.util.concurrent.ArrayBlockingQueue;
|
||||
import java.util.concurrent.BlockingQueue;
|
||||
import java.util.concurrent.ScheduledFuture;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.regex.Matcher;
|
||||
import java.util.regex.Pattern;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.eclipse.jdt.annotation.Nullable;
|
||||
import org.openhab.core.library.types.OnOffType;
|
||||
import org.openhab.core.library.types.StringType;
|
||||
import org.openhab.core.thing.ChannelUID;
|
||||
import org.openhab.core.thing.Thing;
|
||||
import org.openhab.core.thing.ThingStatus;
|
||||
import org.openhab.core.thing.ThingStatusDetail;
|
||||
import org.openhab.core.thing.binding.BaseThingHandler;
|
||||
import org.openhab.core.types.Command;
|
||||
import org.openhab.core.types.RefreshType;
|
||||
import org.openhab.core.types.State;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
/**
|
||||
* The {@link TpLinkRouterHandler} is responsible for handling commands, which are
|
||||
* sent to one of the channels.
|
||||
*
|
||||
* @author Olivier Marceau - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class TpLinkRouterHandler extends BaseThingHandler implements TpLinkRouterTelenetListener {
|
||||
|
||||
private static final long RECONNECT_DELAY = TimeUnit.MINUTES.toMillis(1);
|
||||
|
||||
private static final String REFRESH_CMD = "wlctl show";
|
||||
private static final String WIFI_ON_CMD = "wlctl set --switch on";
|
||||
private static final String WIFI_OFF_CMD = "wlctl set --switch off";
|
||||
private static final String QSS_ON_CMD = "wlctl set --qss on";
|
||||
private static final String QSS_OFF_CMD = "wlctl set --qss off";
|
||||
|
||||
private final Logger logger = LoggerFactory.getLogger(TpLinkRouterHandler.class);
|
||||
|
||||
private final TpLinkRouterTelnetConnector connector = new TpLinkRouterTelnetConnector();
|
||||
private final BlockingQueue<ChannelUIDCommand> commandQueue = new ArrayBlockingQueue<>(1);
|
||||
|
||||
private TpLinkRouterConfiguration config = new TpLinkRouterConfiguration();
|
||||
private @Nullable ScheduledFuture<?> scheduledFuture;
|
||||
|
||||
public TpLinkRouterHandler(Thing thing) {
|
||||
super(thing);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void handleCommand(ChannelUID channelUID, Command command) {
|
||||
if (WIFI_STATUS.equals(channelUID.getId())) {
|
||||
try {
|
||||
commandQueue.put(new ChannelUIDCommand(channelUID, command));
|
||||
} catch (InterruptedException e) {
|
||||
logger.warn("Got exception", e);
|
||||
Thread.currentThread().interrupt();
|
||||
}
|
||||
if (command instanceof RefreshType) {
|
||||
connector.sendCommand(REFRESH_CMD);
|
||||
} else if (command == OnOffType.ON) {
|
||||
connector.sendCommand(WIFI_ON_CMD);
|
||||
} else if (command == OnOffType.OFF) {
|
||||
connector.sendCommand(WIFI_OFF_CMD);
|
||||
}
|
||||
} else if (WIFI_QSS.equals(channelUID.getId())) {
|
||||
try {
|
||||
commandQueue.put(new ChannelUIDCommand(channelUID, command));
|
||||
} catch (InterruptedException e) {
|
||||
logger.warn("Got exception", e);
|
||||
Thread.currentThread().interrupt();
|
||||
}
|
||||
if (command instanceof RefreshType) {
|
||||
connector.sendCommand(REFRESH_CMD);
|
||||
} else if (command == OnOffType.ON) {
|
||||
connector.sendCommand(QSS_ON_CMD);
|
||||
} else if (command == OnOffType.OFF) {
|
||||
connector.sendCommand(QSS_OFF_CMD);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void initialize() {
|
||||
config = getConfigAs(TpLinkRouterConfiguration.class);
|
||||
updateStatus(ThingStatus.UNKNOWN);
|
||||
scheduler.execute(this::createConnection);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void dispose() {
|
||||
ScheduledFuture<?> scheduledFutureLocal = scheduledFuture;
|
||||
if (scheduledFutureLocal != null) {
|
||||
scheduledFutureLocal.cancel(true);
|
||||
scheduledFuture = null;
|
||||
}
|
||||
commandQueue.clear();
|
||||
connector.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
private void createConnection() {
|
||||
connector.dispose();
|
||||
try {
|
||||
connector.connect(this, config, this.getThing().getUID().getAsString());
|
||||
} catch (IOException e) {
|
||||
logger.debug("Error while connecting, will retry in {} ms", RECONNECT_DELAY);
|
||||
scheduler.schedule(this::createConnection, RECONNECT_DELAY, TimeUnit.MILLISECONDS);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void receivedLine(String line) {
|
||||
logger.debug("Received line: {}", line);
|
||||
Pattern pattern = Pattern.compile("(\\w+)=(.+)");
|
||||
Matcher matcher = pattern.matcher(line);
|
||||
if (matcher.find()) {
|
||||
String label = matcher.group(1);
|
||||
String value = matcher.group(2);
|
||||
switch (label) {
|
||||
case "Status":
|
||||
if ("Disabled".equals(value)) {
|
||||
updateState(WIFI_STATUS, OnOffType.OFF);
|
||||
} else if ("Up".equals(value)) {
|
||||
updateState(WIFI_STATUS, OnOffType.ON);
|
||||
} else {
|
||||
logger.warn("Unsupported value {} for label {}", value, label);
|
||||
}
|
||||
break;
|
||||
case "SSID":
|
||||
updateState(WIFI_SSID, StringType.valueOf(value));
|
||||
break;
|
||||
case "bandWidth":
|
||||
updateState(WIFI_BANDWIDTH, StringType.valueOf(value));
|
||||
break;
|
||||
case "QSS":
|
||||
if ("Disabled".equals(value)) {
|
||||
updateState(WIFI_QSS, OnOffType.OFF);
|
||||
} else if ("Enable".equals(value)) {
|
||||
updateState(WIFI_QSS, OnOffType.ON);
|
||||
} else {
|
||||
logger.warn("Unsupported value {} for label {}", value, label);
|
||||
}
|
||||
break;
|
||||
case "SecMode":
|
||||
String[] parts = value.split("\\s|-");
|
||||
updateState(WIFI_SECMODE, StringType.valueOf(parts[0]));
|
||||
updateState(WIFI_AUTHENTICATION, StringType.valueOf(parts[1]));
|
||||
if (parts.length >= 3) {
|
||||
updateState(WIFI_ENCRYPTION, StringType.valueOf(parts[2]));
|
||||
} else {
|
||||
updateState(WIFI_ENCRYPTION, StringType.EMPTY);
|
||||
}
|
||||
break;
|
||||
case "Key":
|
||||
updateState(WIFI_KEY, StringType.valueOf(value));
|
||||
break;
|
||||
default:
|
||||
logger.debug("Unrecognized label {}", label);
|
||||
}
|
||||
} else if ("cmd:SUCC".equals(line)) {
|
||||
ChannelUIDCommand channelUIDCommand = commandQueue.poll();
|
||||
if (channelUIDCommand != null && channelUIDCommand.getCommand() instanceof State) {
|
||||
updateState(channelUIDCommand.getChannelUID(), (State) channelUIDCommand.getCommand());
|
||||
}
|
||||
} else if ("Login incorrect. Try again.".equals(line)) {
|
||||
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, "Login or password incorrect");
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onReaderThreadStopped() {
|
||||
updateStatus(ThingStatus.OFFLINE);
|
||||
logger.debug("try to reconnect in {} ms", RECONNECT_DELAY);
|
||||
scheduler.schedule(this::createConnection, RECONNECT_DELAY, TimeUnit.MILLISECONDS);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onReaderThreadInterrupted() {
|
||||
updateStatus(ThingStatus.OFFLINE);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onReaderThreadStarted() {
|
||||
scheduledFuture = scheduler.scheduleWithFixedDelay(() -> {
|
||||
handleCommand(new ChannelUID(getThing().getUID(), WIFI_STATUS), RefreshType.REFRESH);
|
||||
}, 0, config.refreshInterval, TimeUnit.SECONDS);
|
||||
updateStatus(ThingStatus.ONLINE);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onCommunicationUnavailable() {
|
||||
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
|
||||
"Connection not available. Check if there is not another open connection.");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Stores a command with associated channel
|
||||
*
|
||||
* @author Olivier Marceau - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
class ChannelUIDCommand {
|
||||
private final ChannelUID channelUID;
|
||||
private final Command command;
|
||||
|
||||
public ChannelUIDCommand(ChannelUID channelUID, Command command) {
|
||||
this.channelUID = channelUID;
|
||||
this.command = command;
|
||||
}
|
||||
|
||||
public ChannelUID getChannelUID() {
|
||||
return channelUID;
|
||||
}
|
||||
|
||||
public Command getCommand() {
|
||||
return command;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,55 @@
|
|||
/**
|
||||
* Copyright (c) 2010-2022 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.binding.tplinkrouter.internal;
|
||||
|
||||
import static org.openhab.binding.tplinkrouter.internal.TpLinkRouterBindingConstants.*;
|
||||
|
||||
import java.util.Set;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.eclipse.jdt.annotation.Nullable;
|
||||
import org.openhab.core.thing.Thing;
|
||||
import org.openhab.core.thing.ThingTypeUID;
|
||||
import org.openhab.core.thing.binding.BaseThingHandlerFactory;
|
||||
import org.openhab.core.thing.binding.ThingHandler;
|
||||
import org.openhab.core.thing.binding.ThingHandlerFactory;
|
||||
import org.osgi.service.component.annotations.Component;
|
||||
|
||||
/**
|
||||
* The {@link TpLinkRouterHandlerFactory} is responsible for creating things and thing
|
||||
* handlers.
|
||||
*
|
||||
* @author Olivier Marceau - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
@Component(configurationPid = "binding.tplinkrouter", service = ThingHandlerFactory.class)
|
||||
public class TpLinkRouterHandlerFactory extends BaseThingHandlerFactory {
|
||||
|
||||
private static final Set<ThingTypeUID> SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_ROUTER);
|
||||
|
||||
@Override
|
||||
public boolean supportsThingType(ThingTypeUID thingTypeUID) {
|
||||
return SUPPORTED_THING_TYPES_UIDS.contains(thingTypeUID);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected @Nullable ThingHandler createHandler(Thing thing) {
|
||||
ThingTypeUID thingTypeUID = thing.getThingTypeUID();
|
||||
|
||||
if (THING_TYPE_ROUTER.equals(thingTypeUID)) {
|
||||
return new TpLinkRouterHandler(thing);
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,51 @@
|
|||
/**
|
||||
* Copyright (c) 2010-2022 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.binding.tplinkrouter.internal;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
|
||||
/**
|
||||
* The {@link TpLinkRouterTelenetListener} defines listener for telnet events.
|
||||
*
|
||||
* @author Olivier Marceau - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public interface TpLinkRouterTelenetListener {
|
||||
|
||||
/**
|
||||
* The telnet client has received a line.
|
||||
*
|
||||
* @param line the received line
|
||||
*/
|
||||
void receivedLine(String line);
|
||||
|
||||
/**
|
||||
* The telnet client encountered an IO error.
|
||||
*/
|
||||
void onReaderThreadStopped();
|
||||
|
||||
/**
|
||||
* The telnet client has been interrupted.
|
||||
*/
|
||||
void onReaderThreadInterrupted();
|
||||
|
||||
/**
|
||||
* The telnet client has successfully connected to the receiver.
|
||||
*/
|
||||
void onReaderThreadStarted();
|
||||
|
||||
/**
|
||||
* The telnet socket is unavailable.
|
||||
*/
|
||||
void onCommunicationUnavailable();
|
||||
}
|
|
@ -0,0 +1,164 @@
|
|||
/**
|
||||
* Copyright (c) 2010-2022 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.binding.tplinkrouter.internal;
|
||||
|
||||
import java.io.BufferedReader;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStreamReader;
|
||||
import java.io.InterruptedIOException;
|
||||
import java.io.OutputStreamWriter;
|
||||
import java.net.InetSocketAddress;
|
||||
import java.net.Socket;
|
||||
import java.net.SocketTimeoutException;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
|
||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||
import org.eclipse.jdt.annotation.Nullable;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
/**
|
||||
* The {@link TpLinkRouterTelnetConnector} is responsible for the telnet connection.
|
||||
*
|
||||
* @author Olivier Marceau - Initial contribution
|
||||
*/
|
||||
@NonNullByDefault
|
||||
public class TpLinkRouterTelnetConnector {
|
||||
|
||||
private static final int TIMEOUT_MS = (int) TimeUnit.MINUTES.toMillis(1);
|
||||
|
||||
private final Logger logger = LoggerFactory.getLogger(TpLinkRouterTelnetConnector.class);
|
||||
|
||||
private @Nullable Thread telnetClientThread;
|
||||
private @Nullable Socket socket; // use raw socket since commons net usage seems discouraged
|
||||
private @Nullable OutputStreamWriter out;
|
||||
|
||||
public void connect(TpLinkRouterTelenetListener listener, TpLinkRouterConfiguration config, String thingUID)
|
||||
throws IOException {
|
||||
logger.debug("Connecting to {}", config.hostname);
|
||||
|
||||
Socket socketLocal = new Socket();
|
||||
socketLocal.connect(new InetSocketAddress(config.hostname, config.port));
|
||||
socketLocal.setSoTimeout(TIMEOUT_MS);
|
||||
socketLocal.setKeepAlive(true);
|
||||
|
||||
InputStreamReader inputStream = new InputStreamReader(socketLocal.getInputStream());
|
||||
this.out = new OutputStreamWriter(socketLocal.getOutputStream());
|
||||
this.socket = socketLocal;
|
||||
loginAttempt(listener, inputStream, config);
|
||||
Thread clientThread = new Thread(() -> listenInputStream(listener, inputStream, config));
|
||||
clientThread.setName("OH-binding-" + thingUID);
|
||||
this.telnetClientThread = clientThread;
|
||||
clientThread.start();
|
||||
logger.debug("TP-Link router telnet client connected to {}", config.hostname);
|
||||
}
|
||||
|
||||
public void dispose() {
|
||||
logger.debug("disposing connector");
|
||||
Thread clientThread = telnetClientThread;
|
||||
if (clientThread != null) {
|
||||
clientThread.interrupt();
|
||||
telnetClientThread = null;
|
||||
}
|
||||
Socket socketLocal = socket;
|
||||
if (socketLocal != null) {
|
||||
try {
|
||||
socketLocal.close();
|
||||
} catch (IOException e) {
|
||||
logger.debug("Error while disconnecting telnet client", e);
|
||||
}
|
||||
socket = null;
|
||||
}
|
||||
}
|
||||
|
||||
public void sendCommand(String command) {
|
||||
logger.debug("sending command: {}", command);
|
||||
OutputStreamWriter output = out;
|
||||
if (output != null) {
|
||||
try {
|
||||
output.write(command + '\n');
|
||||
output.flush();
|
||||
} catch (IOException e) {
|
||||
logger.warn("Error sending command", e);
|
||||
}
|
||||
} else {
|
||||
logger.debug("Cannot send command, no telnet connection");
|
||||
}
|
||||
}
|
||||
|
||||
private void listenInputStream(TpLinkRouterTelenetListener listener, InputStreamReader inputStream,
|
||||
TpLinkRouterConfiguration config) {
|
||||
listener.onReaderThreadStarted();
|
||||
BufferedReader in = new BufferedReader(inputStream);
|
||||
try {
|
||||
while (!Thread.currentThread().isInterrupted()) {
|
||||
try {
|
||||
String line = in.readLine();
|
||||
if (line != null && !line.isBlank()) {
|
||||
listener.receivedLine(line);
|
||||
if ("CLI exited after timing out".equals(line)) {
|
||||
OutputStreamWriter output = out;
|
||||
if (output != null) {
|
||||
output.write("\n"); // trigger a "username:" prompt
|
||||
output.flush();
|
||||
loginAttempt(listener, inputStream, config);
|
||||
}
|
||||
}
|
||||
}
|
||||
} catch (SocketTimeoutException e) {
|
||||
logger.trace("Socket timeout");
|
||||
}
|
||||
}
|
||||
} catch (InterruptedIOException e) {
|
||||
logger.debug("Error in telnet connection ", e);
|
||||
} catch (IOException e) {
|
||||
if (!Thread.currentThread().isInterrupted()) {
|
||||
logger.debug("Error in telnet connection ", e);
|
||||
listener.onReaderThreadStopped();
|
||||
}
|
||||
}
|
||||
if (Thread.currentThread().isInterrupted()) {
|
||||
logger.debug("Interrupted client thread");
|
||||
listener.onReaderThreadInterrupted();
|
||||
}
|
||||
}
|
||||
|
||||
private void loginAttempt(TpLinkRouterTelenetListener listener, InputStreamReader inputStreamReader,
|
||||
TpLinkRouterConfiguration config) throws IOException {
|
||||
int charInt;
|
||||
StringBuilder word = new StringBuilder();
|
||||
OutputStreamWriter output = out;
|
||||
if (output != null) {
|
||||
try {
|
||||
while ((charInt = inputStreamReader.read()) != -1) {
|
||||
word.append((char) charInt);
|
||||
logger.trace("received char: {}", (char) charInt);
|
||||
if (word.toString().contains("username:")) {
|
||||
logger.debug("Sending username");
|
||||
output.write(config.username + '\n');
|
||||
output.flush();
|
||||
word = new StringBuilder();
|
||||
}
|
||||
if (word.toString().contains("password:")) {
|
||||
logger.debug("Sending password");
|
||||
output.write(config.password + '\n');
|
||||
output.flush();
|
||||
break;
|
||||
}
|
||||
}
|
||||
} catch (SocketTimeoutException e) {
|
||||
listener.onCommunicationUnavailable();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,9 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<binding:binding id="tplinkrouter" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xmlns:binding="https://openhab.org/schemas/binding/v1.0.0"
|
||||
xsi:schemaLocation="https://openhab.org/schemas/binding/v1.0.0 https://openhab.org/schemas/binding-1.0.0.xsd">
|
||||
|
||||
<name>TpLinkRouter Binding</name>
|
||||
<description>This is the binding for controlling a TP-Link router with telnet.</description>
|
||||
|
||||
</binding:binding>
|
|
@ -0,0 +1,52 @@
|
|||
# binding
|
||||
|
||||
binding.tplinkrouter.name = TpLinkRouter Binding
|
||||
binding.tplinkrouter.description = This is the binding for controlling a TP-Link router with telnet.
|
||||
|
||||
# thing types
|
||||
|
||||
thing-type.tplinkrouter.router.label = Router
|
||||
thing-type.tplinkrouter.router.description = Router device monitored and controlled by telnet connection
|
||||
|
||||
# thing types config
|
||||
|
||||
thing-type.config.tplinkrouter.router.hostname.label = Hostname
|
||||
thing-type.config.tplinkrouter.router.hostname.description = Hostname or IP address of the device
|
||||
thing-type.config.tplinkrouter.router.password.label = Password
|
||||
thing-type.config.tplinkrouter.router.password.description = Password to access the device
|
||||
thing-type.config.tplinkrouter.router.port.label = Port
|
||||
thing-type.config.tplinkrouter.router.port.description = Port for telnet connection
|
||||
thing-type.config.tplinkrouter.router.refreshInterval.label = Refresh Interval
|
||||
thing-type.config.tplinkrouter.router.refreshInterval.description = Interval the device is polled in sec.
|
||||
thing-type.config.tplinkrouter.router.username.label = Username
|
||||
thing-type.config.tplinkrouter.router.username.description = User to access the device
|
||||
|
||||
# channel group types
|
||||
|
||||
channel-group-type.tplinkrouter.wifiGroupType.label = Wifi
|
||||
channel-group-type.tplinkrouter.wifiGroupType.description = Wifi channels
|
||||
|
||||
# channel types
|
||||
|
||||
channel-type.tplinkrouter.authentication.label = Wifi Authentication Mode
|
||||
channel-type.tplinkrouter.authentication.state.option.AUTO = AUTO
|
||||
channel-type.tplinkrouter.authentication.state.option.OPEN = OPEN
|
||||
channel-type.tplinkrouter.authentication.state.option.SHARED = SHARED
|
||||
channel-type.tplinkrouter.authentication.state.option.WPA = WPA
|
||||
channel-type.tplinkrouter.authentication.state.option.WPA2 = WPA2
|
||||
channel-type.tplinkrouter.bandwidth.label = Wifi BandWidth
|
||||
channel-type.tplinkrouter.bandwidth.state.option.Auto = Auto
|
||||
channel-type.tplinkrouter.bandwidth.state.option.20M = 20 MHz
|
||||
channel-type.tplinkrouter.bandwidth.state.option.40M = 40 MHz
|
||||
channel-type.tplinkrouter.encryption.label = Wifi Encryption Mode
|
||||
channel-type.tplinkrouter.encryption.description = Wifi Encryption Mode (only for PSK security mode)
|
||||
channel-type.tplinkrouter.encryption.state.option.AUTO = AUTO
|
||||
channel-type.tplinkrouter.encryption.state.option.TKIP = TKIP
|
||||
channel-type.tplinkrouter.encryption.state.option.AES = AES
|
||||
channel-type.tplinkrouter.key.label = Wifi Key
|
||||
channel-type.tplinkrouter.qss.label = Wifi QSS
|
||||
channel-type.tplinkrouter.security-mode.label = Wifi Security Mode
|
||||
channel-type.tplinkrouter.security-mode.state.option.WEP = WEP
|
||||
channel-type.tplinkrouter.security-mode.state.option.WPA = PSK
|
||||
channel-type.tplinkrouter.ssid.label = Wifi SSID
|
||||
channel-type.tplinkrouter.status.label = Wifi Status
|
|
@ -0,0 +1,86 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<thing:thing-descriptions bindingId="tplinkrouter"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xmlns:thing="https://openhab.org/schemas/thing-description/v1.0.0"
|
||||
xsi:schemaLocation="https://openhab.org/schemas/thing-description/v1.0.0 https://openhab.org/schemas/thing-description-1.0.0.xsd">
|
||||
|
||||
<channel-group-type id="wifiGroupType">
|
||||
<label>Wifi</label>
|
||||
<description>Wifi channels</description>
|
||||
<channels>
|
||||
<channel id="status" typeId="status"/>
|
||||
<channel id="ssid" typeId="ssid"/>
|
||||
<channel id="bandwidth" typeId="bandwidth"/>
|
||||
<channel id="qss" typeId="qss"/>
|
||||
<channel id="secMode" typeId="security-mode"/>
|
||||
<channel id="authentication" typeId="authentication"/>
|
||||
<channel id="encryption" typeId="encryption"/>
|
||||
<channel id="key" typeId="key"/>
|
||||
</channels>
|
||||
</channel-group-type>
|
||||
|
||||
<channel-type id="qss">
|
||||
<item-type>Switch</item-type>
|
||||
<label>Wifi QSS</label>
|
||||
</channel-type>
|
||||
<channel-type id="status">
|
||||
<item-type>Switch</item-type>
|
||||
<label>Wifi Status</label>
|
||||
</channel-type>
|
||||
<channel-type id="ssid">
|
||||
<item-type>String</item-type>
|
||||
<label>Wifi SSID</label>
|
||||
<state pattern="%s" readOnly="true"/>
|
||||
</channel-type>
|
||||
<channel-type id="key">
|
||||
<item-type>String</item-type>
|
||||
<label>Wifi Key</label>
|
||||
<state pattern="%s" readOnly="true"/>
|
||||
</channel-type>
|
||||
<channel-type id="security-mode">
|
||||
<item-type>String</item-type>
|
||||
<label>Wifi Security Mode</label>
|
||||
<state readOnly="true">
|
||||
<options>
|
||||
<option value="WEP">WEP</option>
|
||||
<option value="WPA">PSK</option>
|
||||
</options>
|
||||
</state>
|
||||
</channel-type>
|
||||
<channel-type id="authentication">
|
||||
<item-type>String</item-type>
|
||||
<label>Wifi Authentication Mode</label>
|
||||
<state readOnly="true">
|
||||
<options>
|
||||
<option value="AUTO">AUTO</option>
|
||||
<option value="OPEN">OPEN</option>
|
||||
<option value="SHARED">SHARED</option>
|
||||
<option value="WPA">WPA</option>
|
||||
<option value="WPA2">WPA2</option>
|
||||
</options>
|
||||
</state>
|
||||
</channel-type>
|
||||
<channel-type id="encryption">
|
||||
<item-type>String</item-type>
|
||||
<label>Wifi Encryption Mode</label>
|
||||
<description>Wifi Encryption Mode (only for PSK security mode)</description>
|
||||
<state readOnly="true">
|
||||
<options>
|
||||
<option value="AUTO">AUTO</option>
|
||||
<option value="TKIP">TKIP</option>
|
||||
<option value="AES">AES</option>
|
||||
</options>
|
||||
</state>
|
||||
</channel-type>
|
||||
<channel-type id="bandwidth">
|
||||
<item-type>String</item-type>
|
||||
<label>Wifi BandWidth</label>
|
||||
<state readOnly="true">
|
||||
<options>
|
||||
<option value="Auto">Auto</option>
|
||||
<option value="20M">20 MHz</option>
|
||||
<option value="40M">40 MHz</option>
|
||||
</options>
|
||||
</state>
|
||||
</channel-type>
|
||||
</thing:thing-descriptions>
|
|
@ -0,0 +1,45 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<thing:thing-descriptions bindingId="tplinkrouter"
|
||||
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||
xmlns:thing="https://openhab.org/schemas/thing-description/v1.0.0"
|
||||
xsi:schemaLocation="https://openhab.org/schemas/thing-description/v1.0.0 https://openhab.org/schemas/thing-description-1.0.0.xsd">
|
||||
|
||||
<thing-type id="router">
|
||||
|
||||
<label>Router</label>
|
||||
<description>Router device monitored and controlled by telnet connection</description>
|
||||
|
||||
<channel-groups>
|
||||
<channel-group id="wifi" typeId="wifiGroupType"/>
|
||||
</channel-groups>
|
||||
|
||||
<config-description>
|
||||
<parameter name="hostname" type="text" required="true">
|
||||
<context>network-address</context>
|
||||
<label>Hostname</label>
|
||||
<description>Hostname or IP address of the device</description>
|
||||
</parameter>
|
||||
<parameter name="port" type="integer">
|
||||
<label>Port</label>
|
||||
<description>Port for telnet connection</description>
|
||||
<default>23</default>
|
||||
</parameter>
|
||||
<parameter name="username" type="text" required="true">
|
||||
<label>Username</label>
|
||||
<description>User to access the device</description>
|
||||
</parameter>
|
||||
<parameter name="password" type="text" required="true">
|
||||
<context>password</context>
|
||||
<label>Password</label>
|
||||
<description>Password to access the device</description>
|
||||
</parameter>
|
||||
<parameter name="refreshInterval" type="integer" unit="s" min="1">
|
||||
<label>Refresh Interval</label>
|
||||
<description>Interval the device is polled in sec.</description>
|
||||
<default>60</default>
|
||||
<advanced>true</advanced>
|
||||
</parameter>
|
||||
</config-description>
|
||||
</thing-type>
|
||||
|
||||
</thing:thing-descriptions>
|
|
@ -364,6 +364,7 @@
|
|||
<module>org.openhab.binding.tibber</module>
|
||||
<module>org.openhab.binding.tivo</module>
|
||||
<module>org.openhab.binding.touchwand</module>
|
||||
<module>org.openhab.binding.tplinkrouter</module>
|
||||
<module>org.openhab.binding.tplinksmarthome</module>
|
||||
<module>org.openhab.binding.tr064</module>
|
||||
<module>org.openhab.binding.tradfri</module>
|
||||
|
|
Loading…
Reference in New Issue