diff --git a/pom.xml b/pom.xml index 778f96e..afc233f 100644 --- a/pom.xml +++ b/pom.xml @@ -5,7 +5,7 @@ com.bwssystems.HABridge ha-bridge - 5.2.0RC1 + 5.2.0RC2 jar HA Bridge diff --git a/src/main/java/com/bwssystems/HABridge/dao/DeviceDescriptor.java b/src/main/java/com/bwssystems/HABridge/dao/DeviceDescriptor.java index 1d2b4ee..7d81c8f 100644 --- a/src/main/java/com/bwssystems/HABridge/dao/DeviceDescriptor.java +++ b/src/main/java/com/bwssystems/HABridge/dao/DeviceDescriptor.java @@ -80,7 +80,10 @@ public class DeviceDescriptor{ @SerializedName("deviceState") @Expose private DeviceState deviceState; - + @SerializedName("onFirstDim") + @Expose + private boolean onFirstDim; + public String getName() { return name; } @@ -275,6 +278,14 @@ public class DeviceDescriptor{ this.comments = comments; } + public boolean isOnFirstDim() { + return onFirstDim; + } + + public void setOnFirstDim(boolean onFirstDim) { + this.onFirstDim = onFirstDim; + } + public boolean containsType(String aType) { if(aType == null) return false; diff --git a/src/main/java/com/bwssystems/HABridge/hue/HueMulator.java b/src/main/java/com/bwssystems/HABridge/hue/HueMulator.java index 8c59d40..f45a3ee 100644 --- a/src/main/java/com/bwssystems/HABridge/hue/HueMulator.java +++ b/src/main/java/com/bwssystems/HABridge/hue/HueMulator.java @@ -1096,10 +1096,11 @@ public class HueMulator { DeviceState state = null; Integer targetBri = null; Integer targetBriInc = null; - MultiCommandUtil aMultiUtil = new MultiCommandUtil(); - aMultiUtil.setTheDelay(bridgeSettings.getButtonsleep()); - aMultiUtil.setDelayDefault(bridgeSettings.getButtonsleep()); - aMultiUtil.setSetCount(1); + boolean isColorRequest = false; + boolean isDimRequest = false; + boolean isOnRequest = false; + boolean previousError = false; + ColorData colorData = null; log.debug("hue state change requested: " + userId + " from " + ipAddress + " body: " + body); HueError[] theErrors = bridgeSettingMaster.getBridgeSecurity().validateWhitelistUser(userId, null, bridgeSettingMaster.getBridgeSecurity().isUseLinkButton()); if (theErrors != null) { @@ -1125,136 +1126,127 @@ public class HueMulator { "Could not find device.", "/lights/" + lightId, null, null).getTheErrors(), HueError[].class); } - if (body.contains("\"bri_inc\"")) { - targetBriInc = new Integer(theStateChanges.getBri_inc()); - } - else if (body.contains("\"bri\"")) { - targetBri = new Integer(theStateChanges.getBri()); - } - state = device.getDeviceState(); if (state == null) { state = DeviceState.createDeviceState(device.isColorDevice()); device.setDeviceState(state); } - if (targetBri != null || targetBriInc != null) { - url = device.getDimUrl(); + if (body.contains("\"bri_inc\"")) { + targetBriInc = new Integer(theStateChanges.getBri_inc()); + isDimRequest = true; + } + else if (body.contains("\"bri\"")) { + targetBri = new Integer(theStateChanges.getBri()); + isDimRequest = true; + } - if (url == null || url.length() == 0) - url = device.getOnUrl(); - } else { - if (body.contains("\"xy\"") || body.contains("\"ct\"") || body.contains("\"hue\"")) { - url = device.getColorUrl(); - } else if (theStateChanges.isOn()) { + if (body.contains("\"xy\"") || body.contains("\"ct\"") || body.contains("\"hue\"") || body.contains("\"xy_inc\"") || body.contains("\"ct_inc\"") || body.contains("\"hue_inc\"")) { + List xy = theStateChanges.getXy(); + List xyInc = theStateChanges.getXy_inc(); + Integer ct = theStateChanges.getCt(); + Integer ctInc = theStateChanges.getCt_inc(); + if (xy != null && xy.size() == 2) { + colorData = new ColorData(ColorData.ColorMode.XY, xy); + } else if (xyInc != null && xyInc.size() == 2) { + List current = state.getXy(); + current.set(0, current.get(0) + xyInc.get(0)); + current.set(1, current.get(1) + xyInc.get(1)); + colorData = new ColorData(ColorData.ColorMode.XY, current); + } else if (ct != null && ct != 0) { + colorData = new ColorData(ColorData.ColorMode.CT, ct); + } else if (ctInc != null && ctInc != 0) { + colorData = new ColorData(ColorData.ColorMode.CT, state.getCt() + ctInc); + } + if(colorData != null) + isColorRequest = true; + } + + if (body.contains("\"on\"")) { + isOnRequest = true; + } + + if(isOnRequest || (isDimRequest && device.isOnFirstDim() && !device.getDeviceState().isOn())) { + if (theStateChanges.isOn()) { url = device.getOnUrl(); } else if (!theStateChanges.isOn()) { url = device.getOffUrl(); } - } - // code for backwards compatibility - if(device.getMapType() != null && device.getMapType().equalsIgnoreCase(DeviceMapTypes.HUE_DEVICE[DeviceMapTypes.typeIndex])) { - if(url == null) - url = device.getOnUrl(); - } - if (url != null && !url.equals("")) { - if (!url.startsWith("[")) { - if (url.startsWith("{\"item")) - url = "[" + url + "]"; - else { - if(url.startsWith("{")) - url = "[{\"item\":" + url + "}]"; - else - url = "[{\"item\":\"" + url + "\"}]"; - } - } else if(!url.startsWith("[{\"item\"")) - url = "[{\"item\":" + url + "}]"; - - log.debug("Decode Json for url items: " + url); - CallItem[] callItems = null; - try { - callItems = aGsonHandler.fromJson(url, CallItem[].class); - } catch(JsonSyntaxException e) { - log.warn("Could not decode Json for url items: " + lightId + " for hue state change request: " + userId + " from " - + ipAddress + " body: " + body + " url items: " + url); - return aGsonHandler.toJson(HueErrorResponse.createResponse("3", "/lights/" + lightId, - "Could decode json in request", "/lights/" + lightId, null, null).getTheErrors(), HueError[].class); + // code for backwards compatibility + if(device.getMapType() != null && device.getMapType().equalsIgnoreCase(DeviceMapTypes.HUE_DEVICE[DeviceMapTypes.typeIndex])) { + if(url == null) + url = device.getOnUrl(); } - - for (int i = 0; callItems != null && i < callItems.length; i++) { - if (!ignoreRequester) { - if(!filterByRequester(device.getRequesterAddress(), ipAddress) || !filterByRequester(callItems[i].getFilterIPs(), ipAddress)) { - log.warn("filter for requester address not present in: (device)" + device.getRequesterAddress() + " OR then (item)" + callItems[i].getFilterIPs() + " with request ip of: " + ipAddress); - continue; - } - } - - if (callItems[i].getCount() != null && callItems[i].getCount() > 0) - aMultiUtil.setSetCount(callItems[i].getCount()); - else - aMultiUtil.setSetCount(1); - // code for backwards compatibility - if((callItems[i].getType() == null || callItems[i].getType().trim().isEmpty())) { - if(validMapTypes.validateType(device.getMapType())) - callItems[i].setType(device.getMapType().trim()); - else if(validMapTypes.validateType(device.getDeviceType())) - callItems[i].setType(device.getDeviceType().trim()); - else - callItems[i].setType(DeviceMapTypes.CUSTOM_DEVICE[DeviceMapTypes.typeIndex]); - } - - if (callItems[i].getType() != null) { - for (int x = 0; x < aMultiUtil.getSetCount(); x++) { - if (x > 0 || i > 0) { - try { - Thread.sleep(aMultiUtil.getTheDelay()); - } catch (InterruptedException e) { - // ignore - } - } - if (callItems[i].getDelay() != null && callItems[i].getDelay() > 0) - aMultiUtil.setTheDelay(callItems[i].getDelay()); - else - aMultiUtil.setTheDelay(aMultiUtil.getDelayDefault()); - ColorData colorData = null; - List xy = theStateChanges.getXy(); - List xyInc = theStateChanges.getXy_inc(); - Integer ct = theStateChanges.getCt(); - Integer ctInc = theStateChanges.getCt_inc(); - if (xy != null && xy.size() == 2) { - colorData = new ColorData(ColorData.ColorMode.XY, xy); - } else if (xyInc != null && xyInc.size() == 2) { - List current = state.getXy(); - current.set(0, current.get(0) + xyInc.get(0)); - current.set(1, current.get(1) + xyInc.get(1)); - colorData = new ColorData(ColorData.ColorMode.XY, current); - } else if (ct != null && ct != 0) { - colorData = new ColorData(ColorData.ColorMode.CT, ct); - } else if (ctInc != null && ctInc != 0) { - colorData = new ColorData(ColorData.ColorMode.CT, state.getCt() + ctInc); - } - log.debug("Calling Home device handler for type : " + callItems[i].getType().trim()); - responseString = homeManager.findHome(callItems[i].getType().trim()).deviceHandler(callItems[i], aMultiUtil, lightId, state.getBri(), targetBri, targetBriInc, colorData, device, body); - if(responseString != null && responseString.contains("{\"error\":")) { - x = aMultiUtil.getSetCount(); - } + if (url != null && !url.equals("")) { + responseString = callUrl(url, device, userId, lightId, body, ipAddress, ignoreRequester, targetBri, targetBriInc, colorData); + } else { + log.warn("Could not find on/off url: " + lightId + " for hue state change request: " + userId + " from " + + ipAddress + " body: " + body); + responseString = aGsonHandler.toJson(HueErrorResponse.createResponse("3", "/lights/" + lightId, + "Could not find on/off url.", "/lights/" + lightId, null, null).getTheErrors(), HueError[].class); + previousError = true; + } + } + + if (isDimRequest && !previousError) { + if(!device.isOnFirstDim() ) + url = device.getDimUrl(); + + if (url == null || url.length() == 0) + url = device.getOnUrl(); + + // code for backwards compatibility + if(device.getMapType() != null && device.getMapType().equalsIgnoreCase(DeviceMapTypes.HUE_DEVICE[DeviceMapTypes.typeIndex])) { + if(url == null) + url = device.getOnUrl(); + } + + if (url != null && !url.equals("")) { + if(isOnRequest) { + try { + Thread.sleep(bridgeSettings.getButtonsleep()); + } catch (InterruptedException e) { + // ignore } } - else - log.warn("Call Items type is null <<<" + callItems[i] + ">>>"); + responseString = callUrl(url, device, userId, lightId, body, ipAddress, ignoreRequester, targetBri, targetBriInc, colorData); + } else { + log.warn("Could not find dim url: " + lightId + " for hue state change request: " + userId + " from " + + ipAddress + " body: " + body); + responseString = aGsonHandler.toJson(HueErrorResponse.createResponse("3", "/lights/" + lightId, + "Could not find dim url.", "/lights/" + lightId, null, null).getTheErrors(), HueError[].class); + previousError = true; } - - if(callItems.length == 0) - log.warn("No call items were available: <<<" + url + ">>>"); - } else { - log.warn("Could not find url: " + lightId + " for hue state change request: " + userId + " from " - + ipAddress + " body: " + body); - responseString = aGsonHandler.toJson(HueErrorResponse.createResponse("3", "/lights/" + lightId, - "Could not find url.", "/lights/" + lightId, null, null).getTheErrors(), HueError[].class); } + if (isColorRequest && !previousError) { + url = device.getColorUrl(); + // code for backwards compatibility + if(device.getMapType() != null && device.getMapType().equalsIgnoreCase(DeviceMapTypes.HUE_DEVICE[DeviceMapTypes.typeIndex])) { + if(url == null) + url = device.getOnUrl(); + } + + if (url != null && !url.equals("")) { + if((isOnRequest && !isDimRequest) || isDimRequest) { + try { + Thread.sleep(bridgeSettings.getButtonsleep()); + } catch (InterruptedException e) { + // ignore + } + } + responseString = callUrl(url, device, userId, lightId, body, ipAddress, ignoreRequester, targetBri, targetBriInc, colorData); + } else { + log.warn("Could not find color url: " + lightId + " for hue state change request: " + userId + " from " + + ipAddress + " body: " + body); + responseString = aGsonHandler.toJson(HueErrorResponse.createResponse("3", "/lights/" + lightId, + "Could not find color url.", "/lights/" + lightId, null, null).getTheErrors(), HueError[].class); + previousError = true; + } + } + if (responseString == null || !responseString.contains("[{\"error\":")) { log.debug("Response is in error: " + ((responseString == null) ? "null" : responseString)); if(!device.isNoState()) { @@ -1375,4 +1367,87 @@ public class HueMulator { return aGsonHandler.toJson(theErrors); } + + protected String callUrl(String url, DeviceDescriptor device, String userId, String lightId, String body, String ipAddress, boolean ignoreRequester, Integer targetBri, Integer targetBriInc, ColorData colorData) { + String responseString = null; + MultiCommandUtil aMultiUtil = new MultiCommandUtil(); + aMultiUtil.setTheDelay(bridgeSettings.getButtonsleep()); + aMultiUtil.setDelayDefault(bridgeSettings.getButtonsleep()); + aMultiUtil.setSetCount(1); + + if (!url.startsWith("[")) { + if (url.startsWith("{\"item")) + url = "[" + url + "]"; + else { + if(url.startsWith("{")) + url = "[{\"item\":" + url + "}]"; + else + url = "[{\"item\":\"" + url + "\"}]"; + } + } else if(!url.startsWith("[{\"item\"")) + url = "[{\"item\":" + url + "}]"; + + log.debug("Decode Json for url items: " + url); + CallItem[] callItems = null; + try { + callItems = aGsonHandler.fromJson(url, CallItem[].class); + } catch(JsonSyntaxException e) { + log.warn("Could not decode Json for url items: " + lightId + " for hue state change request: " + userId + " from " + + ipAddress + " body: " + body + " url items: " + url); + return aGsonHandler.toJson(HueErrorResponse.createResponse("3", "/lights/" + lightId, + "Could decode json in request", "/lights/" + lightId, null, null).getTheErrors(), HueError[].class); + } + + for (int i = 0; callItems != null && i < callItems.length; i++) { + if (!ignoreRequester) { + if(!filterByRequester(device.getRequesterAddress(), ipAddress) || !filterByRequester(callItems[i].getFilterIPs(), ipAddress)) { + log.warn("filter for requester address not present in: (device)" + device.getRequesterAddress() + " OR then (item)" + callItems[i].getFilterIPs() + " with request ip of: " + ipAddress); + continue; + } + } + + if (callItems[i].getCount() != null && callItems[i].getCount() > 0) + aMultiUtil.setSetCount(callItems[i].getCount()); + else + aMultiUtil.setSetCount(1); + // code for backwards compatibility + if((callItems[i].getType() == null || callItems[i].getType().trim().isEmpty())) { + if(validMapTypes.validateType(device.getMapType())) + callItems[i].setType(device.getMapType().trim()); + else if(validMapTypes.validateType(device.getDeviceType())) + callItems[i].setType(device.getDeviceType().trim()); + else + callItems[i].setType(DeviceMapTypes.CUSTOM_DEVICE[DeviceMapTypes.typeIndex]); + } + + if (callItems[i].getType() != null) { + for (int x = 0; x < aMultiUtil.getSetCount(); x++) { + if (x > 0 || i > 0) { + try { + Thread.sleep(aMultiUtil.getTheDelay()); + } catch (InterruptedException e) { + // ignore + } + } + if (callItems[i].getDelay() != null && callItems[i].getDelay() > 0) + aMultiUtil.setTheDelay(callItems[i].getDelay()); + else + aMultiUtil.setTheDelay(aMultiUtil.getDelayDefault()); + + log.debug("Calling Home device handler for type : " + callItems[i].getType().trim()); + responseString = homeManager.findHome(callItems[i].getType().trim()).deviceHandler(callItems[i], aMultiUtil, lightId, device.getDeviceState().getBri(), targetBri, targetBriInc, colorData, device, body); + if(responseString != null && responseString.contains("{\"error\":")) { + x = aMultiUtil.getSetCount(); + } + } + } + else + log.warn("Call Items type is null <<<" + callItems[i] + ">>>"); + } + + if(callItems.length == 0) + log.warn("No call items were available: <<<" + url + ">>>"); + + return responseString; + } } diff --git a/src/main/java/com/bwssystems/HABridge/plugins/fhem/FHEMHome.java b/src/main/java/com/bwssystems/HABridge/plugins/fhem/FHEMHome.java index a26a818..d7beb70 100644 --- a/src/main/java/com/bwssystems/HABridge/plugins/fhem/FHEMHome.java +++ b/src/main/java/com/bwssystems/HABridge/plugins/fhem/FHEMHome.java @@ -137,7 +137,7 @@ public class FHEMHome implements Home { List deviceList = new ArrayList(); while(keys.hasNext()) { String key = keys.next(); - theResponse = fhemMap.get(key).getDevices(httpClient); + theResponse = fhemMap.get(key).testGetDevices(httpClient); if(theResponse != null) addFHEMDevices(deviceList, theResponse, key); else { diff --git a/src/main/java/com/bwssystems/HABridge/plugins/fhem/FHEMInstance.java b/src/main/java/com/bwssystems/HABridge/plugins/fhem/FHEMInstance.java index 85a1f72..2bb9a2e 100644 --- a/src/main/java/com/bwssystems/HABridge/plugins/fhem/FHEMInstance.java +++ b/src/main/java/com/bwssystems/HABridge/plugins/fhem/FHEMInstance.java @@ -88,6 +88,275 @@ public class FHEMInstance { return deviceList; } + public List testGetDevices(HTTPHandler httpClient) { + String TestData = "\n" + + "\n" + + " \n" + + " Home, Sweet Home\n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + "
\n" + + "
\n" + + " \n" + + "
\n" + + "
\n" + + "
\n" + + "
\n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + "
\n" + + " \n" + + " \n" + + " \n" + + " \n" + + "
\n" + + "
\n" + + " Save config\n" + + " ?\n" + + "
\n" + + "
\n" + + "
\n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + "
\n" + + "
\n" + + " Alexa\n" + + "
\n" + + "
\n" + + "
\n" + + " System\n" + + "
\n" + + "
\n" + + "
\n" + + " WG-Zimmer\n" + + "
\n" + + "
\n" + + "
\n" + + " habridge\n" + + "
\n" + + "
\n" + + " \n" + + "
\n" + + "
\n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + " \n" + + "
\n" + + "
\n" + + " Logfile\n" + + "
\n" + + "
\n" + + "
\n" + + " Commandref\n" + + "
\n" + + "
\n" + + "
\n" + + " Remote doc\n" + + "
\n" + + "
\n" + + "
\n" + + " Edit files\n" + + "
\n" + + "
\n" + + "
\n" + + " Select style\n" + + "
\n" + + "
\n" + + "
\n" + + " Event monitor\n" + + "
\n" + + "
\n" + + "
\n" + + "
\n" + + "
\n" + + "
\n" + + " \n" + + " \n" + + " \n" + + " \n" + + "
\n" + + "
\n" + + " \n" + + " \n" + + "
\n" + + "
\n" + + "
\n" + + "
\n" + + "
{ \n" + 
+				"  \"Arg\":\"room=habridge\", \n" + 
+				"  \"Results\": [ \n" + 
+				"  { \n" + 
+				"    \"Name\":\"Arbeitslicht\", \n" + 
+				"    \"PossibleSets\":\"on off\", \n" + 
+				"    \"PossibleAttrs\":\"alias comment:textField-long eventMap group room suppressReading userReadings:textField-long verbose:0,1,2,3,4,5 readingList setList useSetExtensions disable disabledForIntervals event-on-change-reading event-on-update-reading event-aggregator event-min-interval stateFormat:textField-long timestamp-on-change-reading alexaName alexaRoom cmdIcon devStateIcon devStateStyle fhem_widget_command fhem_widget_command_2 fhem_widget_command_3 genericDeviceType:security,ignore,switch,outlet,light,blind,thermometer,thermostat,contact,garage,window,lock homebridgeMapping:textField-long icon sortby webCmd widgetOverride userattr\", \n" + 
+				"    \"Internals\": { \n" + 
+				"      \"NAME\": \"Arbeitslicht\", \n" + 
+				"      \"NR\": \"28\", \n" + 
+				"      \"STATE\": \"-\", \n" + 
+				"      \"TYPE\": \"dummy\" \n" + 
+				"    }, \n" + 
+				"    \"Readings\": {      \"state\": { \"Value\":\"on\", \"Time\":\"2017-12-14 15:41:05\" } }, \n" + 
+				"    \"Attributes\": { \n" + 
+				"      \"alexaName\": \"Arbeitslicht\", \n" + 
+				"      \"alexaRoom\": \"alexaroom\", \n" + 
+				"      \"fhem_widget_command\": \"{  \\u0022allowed_values\\u0022 : [    \\u0022on\\u0022  ],  \\u0022order\\u0022 : 0}\", \n" + 
+				"      \"icon\": \"scene_office\", \n" + 
+				"      \"room\": \"Alexa,habridge\", \n" + 
+				"      \"setList\": \"on off\", \n" + 
+				"      \"stateFormat\": \"-\", \n" + 
+				"      \"webCmd\": \"on\" \n" + 
+				"    } \n" + 
+				"  }, \n" + 
+				"  { \n" + 
+				"    \"Name\":\"DeckenlampeLinks\", \n" + 
+				"    \"PossibleSets\":\"on off dim dimup dimdown HSV RGB sync pair unpair\", \n" + 
+				"    \"PossibleAttrs\":\"alias comment:textField-long eventMap group room suppressReading userReadings:textField-long verbose:0,1,2,3,4,5 gamma dimStep defaultColor defaultRamp colorCast whitePoint event-on-change-reading event-on-update-reading event-aggregator event-min-interval stateFormat:textField-long timestamp-on-change-reading alexaName alexaRoom cmdIcon devStateIcon devStateStyle fhem_widget_command fhem_widget_command_2 fhem_widget_command_3 genericDeviceType:security,ignore,switch,outlet,light,blind,thermometer,thermostat,contact,garage,window,lock homebridgeMapping:textField-long icon sortby webCmd widgetOverride \n" + 
+				"                AlleLampen AlleLampen_map\n" + 
+				"                DeckenLampen DeckenLampen_map structexclude userattr\", \n" + 
+				"    \"Internals\": { \n" + 
+				"      \"CONNECTION\": \"bridge-V3\", \n" + 
+				"      \"DEF\": \"RGBW2 bridge-V3:10.2.3.3\", \n" + 
+				"      \"IP\": \"10.2.3.3\", \n" + 
+				"      \"LEDTYPE\": \"RGBW2\", \n" + 
+				"      \"NAME\": \"DeckenlampeLinks\", \n" + 
+				"      \"NR\": \"18\", \n" + 
+				"      \"NTFY_ORDER\": \"50-DeckenlampeLinks\", \n" + 
+				"      \"PORT\": \"8899\", \n" + 
+				"      \"PROTO\": \"0\", \n" + 
+				"      \"SLOT\": \"5\", \n" + 
+				"      \"STATE\": \"off\", \n" + 
+				"      \"TYPE\": \"WifiLight\" \n" + 
+				"    }, \n" + 
+				"    \"Readings\": { \n" + 
+				"      \"RGB\": { \"Value\":\"000000\", \"Time\":\"2017-12-14 15:41:10\" }, \n" + 
+				"      \"brightness\": { \"Value\":\"0\", \"Time\":\"2017-12-14 15:41:10\" }, \n" + 
+				"      \"hue\": { \"Value\":\"0\", \"Time\":\"2017-12-14 15:41:10\" }, \n" + 
+				"      \"saturation\": { \"Value\":\"0\", \"Time\":\"2017-12-14 15:41:10\" }, \n" + 
+				"      \"state\": { \"Value\":\"off\", \"Time\":\"2017-12-14 15:41:10\" } \n" + 
+				"    }, \n" + 
+				"    \"Attributes\": { \n" + 
+				"      \"AlleLampen\": \"AlleLampen\", \n" + 
+				"      \"DeckenLampen\": \"DeckenLampen\", \n" + 
+				"      \"fhem_widget_command\": \"{ \\u0022locations\\u0022 : [ \\u0022APP\\u0022, \\u0022WATCH\\u0022, \\u0022WIDGET\\u0022 ], \\u0022allowed_values\\u0022 : [ \\u0022off\\u0022, \\u0022on\\u0022 ], \\u0022order\\u0022 : 6}\", \n" + 
+				"      \"room\": \"habridge,Alexa,WG-Zimmer\", \n" + 
+				"      \"userattr\": \"AlleLampen AlleLampen_map\n" + 
+				"                DeckenLampen DeckenLampen_map structexclude\", \n" + 
+				"      \"webCmd\": \"RGB\", \n" + 
+				"      \"widgetOverride\": \"RGB:colorpicker,RGB\" \n" + 
+				"    } \n" + 
+				"  }  ], \n" + 
+				"  \"totalResultsReturned\":2 \n" + 
+				"}\n" + 
+				"            
\n" + + "
\n" + + " \n" + + ""; + + String TestData2 = "
\n" + + "
\n" + "{   \"Arg\":\"room=HaBridge\",   \"Results\": [   {     \"Name\":\"wifi_steckdose3\",     \"PossibleSets\":\"on:noArg off:noArg off on toggle\",     \"PossibleAttrs\":\"alias comment:textField-long eventMap group room suppressReading userReadings:textField-long verbose:0,1,2,3,4,5 IODev qos retain publishSet publishSet_.* subscribeReading_.* autoSubscribeReadings event-on-change-reading event-on-update-reading event-aggregator event-min-interval stateFormat:textField-long timestamp-on-change-reading alarmDevice:Actor,Sensor alarmSettings cmdIcon devStateIcon devStateStyle icon lightSceneParamsToSave lightSceneRestoreOnlyIfChanged:1,0 sortby structexclude webCmd webCmdLabel:textField-long widgetOverride userattr\",     \"Internals\": {       \"CHANGED\": \"null\",       \"NAME\": \"wifi_steckdose3\",       \"NR\": \"270\",       \"STATE\": \"off\",       \"TYPE\": \"MQTT_DEVICE\",       \"retain\": \"*:1 \"     },     \"Readings\": {       \"state\": { \"Value\":\"OFF\", \"Time\":\"2018-01-01 23:01:21\" },       \"transmission-state\": { \"Value\":\"subscription acknowledged\", \"Time\":\"2018-01-03 22:34:00\" }     },     \"Attributes\": {       \"IODev\": \"myBroker\",       \"alias\": \"Stecki\",       \"devStateIcon\": \"on:black_Steckdose.on off:black_Steckdose.off\",       \"event-on-change-reading\": \"state\",       \"eventMap\": \"ON:on OFF:off\",       \"publishSet\": \"on off toggle /SmartHome/az/stecker/cmnd/POWER\",       \"retain\": \"1\",       \"room\": \"HaBridge,Arbeitszimmer,mqtt\",       \"stateFormat\": \"state\",       \"subscribeReading_state\": \"/SmartHome/az/stecker/stat/POWER\",       \"webCmd\": \"on:off:toggle\"     }   }  ],   \"totalResultsReturned\":1 }" + 
+				"            
\n" + + "
\n" + + " \n" + + ""; + List deviceList = null; + FHEMItem theFhemStates; + String theUrl = null; + String theData; + NameValue[] headers = null; + if(theFhem.getSecure() != null && theFhem.getSecure()) + theUrl = "https://"; + else + theUrl = "http://"; + if(theFhem.getUsername() != null && !theFhem.getUsername().isEmpty() && theFhem.getPassword() != null && !theFhem.getPassword().isEmpty()) { + theUrl = theUrl + theFhem.getUsername() + ":" + theFhem.getPassword() + "@"; + } + theUrl = theUrl + theFhem.getIp() + ":" + theFhem.getPort() + "/fhem?cmd=jsonlist2"; + if(theFhem.getWebhook() != null && !theFhem.getWebhook().trim().isEmpty()) + theUrl = theUrl + "%20room=" + theFhem.getWebhook().trim(); +// theData = httpClient.doHttpRequest(theUrl, HttpGet.METHOD_NAME, "application/json", null, headers); + theData = TestData; + if(theData != null) { + log.debug("GET FHEM States - data: " + theData); + theData = getJSONData(theData); + theFhemStates = new Gson().fromJson(theData, FHEMItem.class); + if(theFhemStates == null) { + log.warn("Cannot get any devices for FHEM " + theFhem.getName() + " as response is not parsable."); + } + else { + deviceList = new ArrayList(); + + for (Result aResult:theFhemStates.getResults()) { + FHEMDevice aNewFhemDeviceDevice = new FHEMDevice(); + aNewFhemDeviceDevice.setItem(aResult); + aNewFhemDeviceDevice.setAddress(theFhem.getIp() + ":" + theFhem.getPort()); + aNewFhemDeviceDevice.setName(theFhem.getName()); + deviceList.add(aNewFhemDeviceDevice); + } + } + } + else + log.warn("Cannot get an devices for FHEM " + theFhem.getName() + " http call failed."); + return deviceList; + } + public String getJSONData(String response) { String theData; theData = response.substring(response.indexOf("
") + 4);
diff --git a/src/main/java/com/bwssystems/HABridge/plugins/http/HTTPHandler.java b/src/main/java/com/bwssystems/HABridge/plugins/http/HTTPHandler.java
index 195076a..6484be4 100644
--- a/src/main/java/com/bwssystems/HABridge/plugins/http/HTTPHandler.java
+++ b/src/main/java/com/bwssystems/HABridge/plugins/http/HTTPHandler.java
@@ -103,9 +103,10 @@ public class HTTPHandler {
 					retryCount = 2;
 				} else if (response != null) {
 					log.warn("HTTP response code was not an expected successful response of between 200 - 299, the code was: "
-									+ response.getStatusLine());
+									+ response.getStatusLine() + " with the content of <<<" + theContent + ">>>");
 					if (response.getStatusLine().getStatusCode() == 504) {
 						log.warn("HTTP response code was 504, retrying...");
+						log.debug("The 504 error content is <<<" + theContent + ">>>");
 					} else
 						retryCount = 2;
 					
diff --git a/src/main/java/com/bwssystems/HABridge/plugins/somfy/SomfyInfo.java b/src/main/java/com/bwssystems/HABridge/plugins/somfy/SomfyInfo.java
index 4b74c6e..be1d862 100644
--- a/src/main/java/com/bwssystems/HABridge/plugins/somfy/SomfyInfo.java
+++ b/src/main/java/com/bwssystems/HABridge/plugins/somfy/SomfyInfo.java
@@ -75,7 +75,7 @@ public class SomfyInfo {
 		urlEncodedFormEntity.writeTo(bos);
 		String body = bos.toString();
 		String response = httpClient.doHttpRequest(BASE_URL + "json/login",HttpPost.METHOD_NAME, "application/x-www-form-urlencoded", body,httpHeader);
-		log.debug(response);
+		log.debug("Somfy login response <<<" + response + ">>>");
 	}
 
 	private NameValue[] getHttpHeaders() {
@@ -89,7 +89,7 @@ public class SomfyInfo {
 		NameValue[] httpHeader = getHttpHeaders();
 		log.info("Making SOMFY http setup call");
 		String response = httpClient.doHttpRequest(BASE_URL + "json/getSetup", HttpGet.METHOD_NAME, "", "", httpHeader );
-		log.debug(response);
+		log.debug("Somfy getSetup response <<<" + response + ">>>");
 		GetSetup setupData = new Gson().fromJson(response, GetSetup.class);
 		return setupData;
 	}
@@ -98,7 +98,7 @@ public class SomfyInfo {
 		login(namedIP.getUsername(), namedIP.getPassword());
 		log.info("Making SOMFY http exec call");
 		String response = httpClient.doHttpRequest(BASE_URL_ENDUSER + "exec/apply", HttpPost.METHOD_NAME, "application/json;charset=UTF-8", jsonToPost, getHttpHeaders());
-		log.info(response);
+		log.debug("Somfy exec reply response <<<" + response + ">>>");
 	}
 
 
diff --git a/src/main/resources/public/css/colorpicker.min.css b/src/main/resources/public/css/colorpicker.min.css
new file mode 100644
index 0000000..92a57d8
--- /dev/null
+++ b/src/main/resources/public/css/colorpicker.min.css
@@ -0,0 +1 @@
+.colorpicker-visible,.colorpicker-visible .dropdown-menu{display:block!important}colorpicker-saturation{display:block;width:100px;height:100px;background-image:url();background-size:contain;cursor:crosshair;float:left}colorpicker-saturation i{display:block;height:7px;width:7px;border:1px solid #000;border-radius:5px;position:absolute;top:0;left:0;margin:-4px 0 0 -4px}colorpicker-saturation i::after{content:'';display:block;height:7px;width:7px;border:1px solid #fff;border-radius:5px}colorpicker-alpha,colorpicker-hue{width:15px;height:100px;float:left;cursor:row-resize;margin-left:4px;margin-bottom:4px}colorpicker-alpha i,colorpicker-hue i{display:block;height:2px;background:#000;border-top:1px solid #fff;position:absolute;top:0;left:0;width:100%;margin-top:-1px}.colorpicker,colorpicker-alpha{display:none}colorpicker-hue{background-image:url();background-size:contain}.colorpicker-color,colorpicker-alpha{background-image:url();background-size:10px 100%}.colorpicker{top:0;left:0;z-index:99999}.colorpicker colorpicker-alpha,.colorpicker colorpicker-hue,.colorpicker colorpicker-saturation{position:relative}.colorpicker input{width:100px;font-size:11px;color:#000;background-color:#fff}.colorpicker.alpha{min-width:140px}.colorpicker.alpha colorpicker-alpha{display:block}.colorpicker.dropdown{position:absolute}.colorpicker.colorpicker-fixed-position{position:fixed}.colorpicker .dropdown-menu::after,.colorpicker .dropdown-menu::before{content:'';display:inline-block;position:absolute}.colorpicker .dropdown-menu::after{clear:both;border:6px solid transparent;top:-5px;left:7px}.colorpicker .dropdown-menu::before{border:7px solid transparent;top:-6px;left:6px}.colorpicker .dropdown-menu{position:static;top:0;left:0;min-width:129px;padding:4px;margin-top:0}.colorpicker-position-top .dropdown-menu::after{border-top:6px solid #fff;border-bottom:0;top:auto;bottom:-5px}.colorpicker-position-top .dropdown-menu::before{border-top:7px solid rgba(0,0,0,.2);border-bottom:0;top:auto;bottom:-6px}.colorpicker-position-right .dropdown-menu::after{border-right:6px solid #fff;border-left:0;top:11px;left:-5px}.colorpicker-position-right .dropdown-menu::before{border-right:7px solid rgba(0,0,0,.2);border-left:0;top:10px;left:-6px}.colorpicker-position-bottom .dropdown-menu::after{border-bottom:6px solid #fff;border-top:0}.colorpicker-position-bottom .dropdown-menu::before{border-bottom:7px solid rgba(0,0,0,.2);border-top:0}.colorpicker-position-left .dropdown-menu::after{border-left:6px solid #fff;border-right:0;top:11px;left:auto;right:-5px}.colorpicker-position-left .dropdown-menu::before{border-left:7px solid rgba(0,0,0,.2);border-right:0;top:10px;left:auto;right:-6px}colorpicker-preview{display:block;height:10px;margin:5px 0 3px;clear:both;background-position:0 100%}
\ No newline at end of file
diff --git a/src/main/resources/public/img/alpha.png b/src/main/resources/public/img/alpha.png
new file mode 100644
index 0000000..eaddb40
Binary files /dev/null and b/src/main/resources/public/img/alpha.png differ
diff --git a/src/main/resources/public/img/hue.png b/src/main/resources/public/img/hue.png
new file mode 100644
index 0000000..ad35ccb
Binary files /dev/null and b/src/main/resources/public/img/hue.png differ
diff --git a/src/main/resources/public/img/saturation.png b/src/main/resources/public/img/saturation.png
new file mode 100644
index 0000000..82a4e3d
Binary files /dev/null and b/src/main/resources/public/img/saturation.png differ
diff --git a/src/main/resources/public/index.html b/src/main/resources/public/index.html
index 6d23c7e..afc97f4 100644
--- a/src/main/resources/public/index.html
+++ b/src/main/resources/public/index.html
@@ -15,6 +15,7 @@
 	
 	
 	
+