/**
* Copyright (c) 2010-2016 by the respective copyright holders.
*
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*/
package org.openhab.binding.mpd.internal;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Set;
import org.apache.commons.lang.StringUtils;
import org.openhab.binding.mpd.MpdBindingProvider;
import org.openhab.core.binding.BindingConfig;
import org.openhab.core.items.Item;
import org.openhab.core.library.items.DimmerItem;
import org.openhab.core.library.items.NumberItem;
import org.openhab.core.library.items.StringItem;
import org.openhab.core.library.items.SwitchItem;
import org.openhab.model.item.binding.AbstractGenericBindingProvider;
import org.openhab.model.item.binding.BindingConfigParseException;
/**
* <p>
* This class can parse information from the generic binding format and
* provides MPD binding information from it. It registers as a
* {@link MpdBindingProvider} service as well.
* </p>
*
* <p>
* Here are some examples for valid binding configuration strings:
* <ul>
* <li><code>{ mpd="ON:bath:play, OFF:bath:stop" }</code> - starts or stops playing the player named 'bath'</li>
* <li><code>{ mpd="INCREASE:bath:volume_increase, DECREASE:bath:volume_decrease" }</code> - increases or decreases the
* volume of the player named 'bath'</li>
* </ul>
*
* @author Thomas.Eichstaedt-Engelen
* @author Matthew Bowman
*
* @since 0.8.0
*/
public class MpdGenericBindingProvider extends AbstractGenericBindingProvider implements MpdBindingProvider {
private static final String PARAM_SUFFIX = ":param";
/**
* {@inheritDoc}
*/
@Override
public String getBindingType() {
return "mpd";
}
/**
* @{inheritDoc}
*/
@Override
public void validateItemType(Item item, String bindingConfig) throws BindingConfigParseException {
if (!(item instanceof SwitchItem || item instanceof DimmerItem || item instanceof StringItem
|| item instanceof NumberItem)) {
throw new BindingConfigParseException(
"item '" + item.getName() + "' is of type '" + item.getClass().getSimpleName()
+ "', only Switch- and DimmerItems are allowed - please check your *.items configuration");
}
}
/**
* {@inheritDoc}
*/
@Override
public void processBindingConfiguration(String context, Item item, String bindingConfig)
throws BindingConfigParseException {
super.processBindingConfiguration(context, item, bindingConfig);
MpdBindingConfig config = new MpdBindingConfig();
parseBindingConfig(bindingConfig, config);
addBindingConfig(item, config);
}
protected void parseBindingConfig(String bindingConfigs, MpdBindingConfig config)
throws BindingConfigParseException {
String bindingConfig = StringUtils.substringBefore(bindingConfigs, ",");
String bindingConfigTail = StringUtils.substringAfter(bindingConfigs, ",");
String[] configParts = bindingConfig.split(":");
if (configParts.length != 3) {
throw new BindingConfigParseException(
"MPD binding configuration must consist of three parts [config=" + configParts + "]");
}
String command = StringUtils.trim(configParts[0]);
String playerId = StringUtils.trim(configParts[1]);
String playerCommand = StringUtils.trim(configParts[2]);
// check for optional command=param binding
String[] playerCommandParts = playerCommand.split("=");
if (playerCommandParts.length == 2) {
// rewrite command=param -> command
playerCommand = StringUtils.trim(playerCommandParts[0]);
String playerCommandParam = StringUtils.trim(playerCommandParts[1]);
// save the param in the config
config.put(command + PARAM_SUFFIX, playerCommandParam);
}
// if there are more commands to parse do that recursively ...
if (StringUtils.isNotBlank(bindingConfigTail)) {
parseBindingConfig(bindingConfigTail, config);
}
config.put(command, playerId + ":" + playerCommand);
}
/**
* {@inheritDoc}
*/
@Override
public String getPlayerCommand(String itemName, String command) {
MpdBindingConfig config = (MpdBindingConfig) bindingConfigs.get(itemName);
return config != null ? config.get(command) : null;
}
/**
* {@inheritDoc}
*/
@Override
public String getPlayerCommandParam(String itemName, String command) {
MpdBindingConfig config = (MpdBindingConfig) bindingConfigs.get(itemName);
return config != null ? config.get(command + PARAM_SUFFIX) : null;
}
/**
* {@inheritDoc}
*/
@Override
public String[] getItemNamesByPlayerAndPlayerCommand(String playerId, PlayerCommandTypeMapping playerCommand) {
Set<String> itemNames = new HashSet<String>();
for (String itemName : bindingConfigs.keySet()) {
MpdBindingConfig mpdConfig = (MpdBindingConfig) bindingConfigs.get(itemName);
String key;
switch (playerCommand) {
case VOLUME:
key = "PERCENT";
break;
case TRACKINFO:
key = "TITLE";
break;
case TRACKARTIST:
key = "ARTIST";
break;
case PLAYSONGID:
key = "NUMBER";
default:
key = playerCommand.type.toString();
break;
}
if (mpdConfig.containsKey(key)) {
// we check to make sure the binding config contains
// playerId:playerCommand otherwise we get extra items
String actual = mpdConfig.get(key);
String expected = playerId + ":" + playerCommand.getPlayerCommand();
if (StringUtils.equals(actual, expected)) {
itemNames.add(itemName);
}
}
}
return itemNames.toArray(new String[itemNames.size()]);
}
/**
* {@inheritDoc}
*/
@Override
public String[] getItemNamesByPlayerOutputCommand(String playerId, PlayerCommandTypeMapping command, int outputId) {
Set<String> itemNames = new HashSet<String>();
for (String itemName : bindingConfigs.keySet()) {
MpdBindingConfig config = (MpdBindingConfig) bindingConfigs.get(itemName);
// We're looking for either...
// ---
// ON = <player-id>:enable
// ON:param = <output-id>
// --- or ---
// OFF = <player-id>:disable
// OFF:param = <output-id>
// ---
String k1 = command.type.toString();
String v1 = playerId + ":" + command.toString().toLowerCase();
String k2 = command.type.toString() + PARAM_SUFFIX;
String v2 = String.valueOf(outputId);
if (StringUtils.equals(config.get(k1), v1) && StringUtils.equals(config.get(k2), v2)) {
itemNames.add(itemName);
}
}
return itemNames.toArray(new String[itemNames.size()]);
}
/**
* This is an internal data structure to store information from the binding
* config strings and use it to answer the requests to the MPD binding
* provider.
*/
static class MpdBindingConfig extends HashMap<String, String> implements BindingConfig {
/** generated serialVersion UID */
private static final long serialVersionUID = 6164971643530954095L;
}
}