org.openhab.binding.denonmarantz.internal.connector.telnet.DenonMarantzTelnetConnector.java Source code

Java tutorial

Introduction

Here is the source code for org.openhab.binding.denonmarantz.internal.connector.telnet.DenonMarantzTelnetConnector.java

Source

/**
 * Copyright (c) 2010-2019 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.denonmarantz.internal.connector.telnet;

import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.concurrent.Future;
import java.util.concurrent.ScheduledExecutorService;
import java.util.regex.Pattern;

import org.apache.commons.lang.StringUtils;
import org.openhab.binding.denonmarantz.internal.DenonMarantzState;
import org.openhab.binding.denonmarantz.internal.config.DenonMarantzConfiguration;
import org.openhab.binding.denonmarantz.internal.connector.DenonMarantzConnector;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

/**
 * This class makes the connection to the receiver and manages it.
 * It is also responsible for sending commands to the receiver.
 *
 * @author Jeroen Idserda - Initial Contribution (1.x Binding)
 * @author Jan-Willem Veldhuis - Refactored for 2.x
 */
public class DenonMarantzTelnetConnector extends DenonMarantzConnector implements DenonMarantzTelnetListener {

    private final Logger logger = LoggerFactory.getLogger(DenonMarantzTelnetConnector.class);

    // All regular commands. Example: PW, SICD, SITV, Z2MU
    private static final Pattern COMMAND_PATTERN = Pattern.compile("^([A-Z0-9]{2})(.+)$");

    // Example: E2Counting Crows
    private static final Pattern DISPLAY_PATTERN = Pattern.compile("^(E|A)([0-9]{1})(.+)$");

    private static final BigDecimal NINETYNINE = new BigDecimal("99");

    private DenonMarantzTelnetClient telnetClient;

    private boolean displayNowplaying = false;

    protected boolean disposing = false;

    private Future<?> telnetStateRequest;

    private Future<?> telnetRunnable;

    public DenonMarantzTelnetConnector(DenonMarantzConfiguration config, DenonMarantzState state,
            ScheduledExecutorService scheduler) {
        this.config = config;
        this.scheduler = scheduler;
        this.state = state;
    }

    /**
     * Set up the connection to the receiver. Either using Telnet or by polling the HTTP API.
     */
    @Override
    public void connect() {
        telnetClient = new DenonMarantzTelnetClient(config, this);
        telnetRunnable = scheduler.submit(telnetClient);
    }

    @Override
    public void telnetClientConnected(boolean connected) {
        if (!connected) {
            if (config.isTelnet() && !disposing) {
                logger.debug("Telnet client disconnected.");
                state.connectionError(
                        "Error connecting to the telnet port. Consider disabling telnet in this Thing's configuration to use HTTP polling instead.");
            }
        } else {
            refreshState();
        }
    }

    /**
     * Shutdown the telnet client (if initialized) and the http client
     */
    @Override
    public void dispose() {
        logger.debug("disposing connector");
        disposing = true;

        if (telnetStateRequest != null) {
            telnetStateRequest.cancel(true);
            telnetStateRequest = null;
        }

        if (telnetClient != null && !telnetRunnable.isDone()) {
            telnetRunnable.cancel(true);
            telnetClient.shutdown();
        }
    }

    private void refreshState() {
        // Sends a series of state query commands over the telnet connection
        telnetStateRequest = scheduler.submit(() -> {
            ArrayList<String> cmds = new ArrayList<String>(Arrays.asList("PW?", "MS?", "MV?", "ZM?", "MU?", "SI?"));
            if (config.getZoneCount() > 1) {
                cmds.add("Z2?");
                cmds.add("Z2MU?");
            }
            if (config.getZoneCount() > 2) {
                cmds.add("Z3?");
                cmds.add("Z3MU?");
            }
            for (String cmd : cmds) {
                internalSendCommand(cmd);
                try {
                    Thread.sleep(300);
                } catch (InterruptedException e) {
                    logger.trace("requestStateOverTelnet() - Interrupted while requesting state.");
                }
            }
        });
    }

    /**
     * This method tries to parse information received over the telnet connection.
     * It can be quite unreliable. Some chars go missing or turn into other chars. That's
     * why each command is validated using a regex.
     *
     * @param line The received command (one line)
     */
    @Override
    public void receivedLine(String line) {
        if (COMMAND_PATTERN.matcher(line).matches()) {
            /*
             * This splits the commandString into the command and the parameter. SICD
             * for example has SI as the command and CD as the parameter.
             */
            String command = line.substring(0, 2);
            String value = line.substring(2, line.length()).trim();

            logger.debug("Received Command: {}, value: {}", command, value);

            // use received command (event) from telnet to update state
            switch (command) {
            case "SI": // Switch Input
                state.setInput(value);
                break;
            case "PW": // Power
                if (value.equals("ON") || value.equals("STANDBY")) {
                    state.setPower(value.equals("ON"));
                }
                break;
            case "MS": // Main zone surround program
                state.setSurroundProgram(value);
                break;
            case "MV": // Main zone volume
                if (StringUtils.isNumeric(value)) {
                    state.setMainVolume(fromDenonValue(value));
                }
                break;
            case "MU": // Main zone mute
                if (value.equals("ON") || value.equals("OFF")) {
                    state.setMute(value.equals("ON"));
                }
                break;
            case "NS": // Now playing information
                processTitleCommand(value);
                break;
            case "Z2": // Zone 2
                if (value.equals("ON") || value.equals("OFF")) {
                    state.setZone2Power(value.equals("ON"));
                } else if (value.equals("MUON") || value.equals("MUOFF")) {
                    state.setZone2Mute(value.equals("MUON"));
                } else if (StringUtils.isNumeric(value)) {
                    state.setZone2Volume(fromDenonValue(value));
                } else {
                    state.setZone2Input(value);
                }
                break;
            case "Z3": // Zone 3
                if (value.equals("ON") || value.equals("OFF")) {
                    state.setZone3Power(value.equals("ON"));
                } else if (value.equals("MUON") || value.equals("MUOFF")) {
                    state.setZone3Mute(value.equals("MUON"));
                } else if (StringUtils.isNumeric(value)) {
                    state.setZone3Volume(fromDenonValue(value));
                } else {
                    state.setZone3Input(value);
                }
                break;
            case "ZM": // Main zone
                if (value.equals("ON") || value.equals("OFF")) {
                    state.setMainZonePower(value.equals("ON"));
                }
                break;
            }
        } else {
            logger.trace("Ignoring received line: '{}'", line);
        }
    }

    private BigDecimal fromDenonValue(String string) {
        /*
         * 455 = 45,5
         * 45 = 45
         * 045 = 4,5
         * 04 = 4
         */
        BigDecimal value = new BigDecimal(string);
        if (value.compareTo(NINETYNINE) == 1 || (string.startsWith("0") && string.length() > 2)) {
            value = value.divide(BigDecimal.TEN);
        }
        return value;
    }

    private void processTitleCommand(String value) {
        if (DISPLAY_PATTERN.matcher(value).matches()) {
            Integer commandNo = Integer.valueOf(value.substring(1, 2));
            String titleValue = value.substring(2);

            if (commandNo == 0) {
                displayNowplaying = titleValue.contains("Now Playing");
            }

            String nowPlaying = displayNowplaying ? cleanupDisplayInfo(titleValue) : "";

            switch (commandNo) {
            case 1:
                state.setNowPlayingTrack(nowPlaying);
                break;
            case 2:
                state.setNowPlayingArtist(nowPlaying);
                break;
            case 4:
                state.setNowPlayingAlbum(nowPlaying);
                break;
            }
        }
    }

    @Override
    protected void internalSendCommand(String command) {
        logger.debug("Sending command '{}'", command);
        if (StringUtils.isBlank(command)) {
            logger.warn("Trying to send empty command");
            return;
        }
        telnetClient.sendCommand(command);
    }

    /**
     * Display info could contain some garbled text, attempt to clean it up.
     */
    private String cleanupDisplayInfo(String titleValue) {
        byte firstByteRemoved[] = Arrays.copyOfRange(titleValue.getBytes(), 1, titleValue.getBytes().length);
        return new String(firstByteRemoved).replaceAll("[\u0000-\u001f]", "");
    }
}