From 9c527abaf159663dfe47695f263e5dcf3201f169 Mon Sep 17 00:00:00 2001 From: lolodomo Date: Mon, 13 Jul 2020 23:06:45 +0200 Subject: [PATCH] [hue] Thing actions still working after a bundle restart (#8115) Related to openhab/openhab-core#1265 Signed-off-by: Laurent Garnier --- .../hue/internal/action/ILightActions.java | 29 ++++++++++++++ .../hue/internal/action/LightActions.java | 40 +++++++++++++++---- 2 files changed, 61 insertions(+), 8 deletions(-) create mode 100644 bundles/org.openhab.binding.hue/src/main/java/org/openhab/binding/hue/internal/action/ILightActions.java diff --git a/bundles/org.openhab.binding.hue/src/main/java/org/openhab/binding/hue/internal/action/ILightActions.java b/bundles/org.openhab.binding.hue/src/main/java/org/openhab/binding/hue/internal/action/ILightActions.java new file mode 100644 index 0000000000000..9a83ba9f8bade --- /dev/null +++ b/bundles/org.openhab.binding.hue/src/main/java/org/openhab/binding/hue/internal/action/ILightActions.java @@ -0,0 +1,29 @@ +/** + * Copyright (c) 2010-2020 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.hue.internal.action; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.eclipse.smarthome.core.library.types.DecimalType; +import org.eclipse.smarthome.core.types.Command; + +/** + * The {@link ILightActions} defines the interface for all thing actions supported by the binding. + * + * @author Laurent Garnier - Initial contribution + */ +@NonNullByDefault +public interface ILightActions { + + public void fadingLightCommand(@Nullable String channel, @Nullable Command command, @Nullable DecimalType fadeTime); +} diff --git a/bundles/org.openhab.binding.hue/src/main/java/org/openhab/binding/hue/internal/action/LightActions.java b/bundles/org.openhab.binding.hue/src/main/java/org/openhab/binding/hue/internal/action/LightActions.java index 57fd8a12bbb0c..20a06759fe34c 100644 --- a/bundles/org.openhab.binding.hue/src/main/java/org/openhab/binding/hue/internal/action/LightActions.java +++ b/bundles/org.openhab.binding.hue/src/main/java/org/openhab/binding/hue/internal/action/LightActions.java @@ -12,6 +12,9 @@ */ package org.openhab.binding.hue.internal.action; +import java.lang.reflect.Method; +import java.lang.reflect.Proxy; + import org.eclipse.jdt.annotation.NonNullByDefault; import org.eclipse.jdt.annotation.Nullable; import org.eclipse.smarthome.core.library.types.DecimalType; @@ -26,14 +29,19 @@ import org.slf4j.LoggerFactory; /** - * This is the automation engine action handler service for the - * fadingLightCommand action. + * The {@link LightActions} defines the thing actions for the hue binding. + *

+ * Note:The static method invokeMethodOf handles the case where + * the test actions instanceof LightActions fails. This test can fail + * due to an issue in openHAB core v2.5.0 where the {@link LightActions} class + * can be loaded by a different classloader than the actions instance. * * @author Jochen Leopold - Initial contribution + * @author Laurent Garnier - new method invokeMethodOf + interface ILightActions */ @ThingActionsScope(name = "hue") @NonNullByDefault -public class LightActions implements ThingActions { +public class LightActions implements ThingActions, ILightActions { private final Logger logger = LoggerFactory.getLogger(LightActions.class); private @Nullable HueLightHandler handler; @@ -47,6 +55,7 @@ public void setThingHandler(@Nullable ThingHandler handler) { return this.handler; } + @Override @RuleAction(label = "@text/actionLabel", description = "@text/actionDesc") public void fadingLightCommand( @ActionInput(name = "channel", label = "@text/actionInputChannelLabel", description = "@text/actionInputChannelDesc") @Nullable String channel, @@ -76,12 +85,27 @@ public void fadingLightCommand( logger.debug("send LightAction to {} with {}ms of fadeTime", channel, fadeTime); } + private static ILightActions invokeMethodOf(@Nullable ThingActions actions) { + if (actions == null) { + throw new IllegalArgumentException("actions cannot be null"); + } + if (actions.getClass().getName().equals(LightActions.class.getName())) { + if (actions instanceof ILightActions) { + return (ILightActions) actions; + } else { + return (ILightActions) Proxy.newProxyInstance(ILightActions.class.getClassLoader(), + new Class[] { ILightActions.class }, (Object proxy, Method method, Object[] args) -> { + Method m = actions.getClass().getDeclaredMethod(method.getName(), + method.getParameterTypes()); + return m.invoke(actions, args); + }); + } + } + throw new IllegalArgumentException("Actions is not an instance of LightActions"); + } + public static void fadingLightCommand(@Nullable ThingActions actions, @Nullable String channel, @Nullable Command command, @Nullable DecimalType fadeTime) { - if (actions instanceof LightActions) { - ((LightActions) actions).fadingLightCommand(channel, command, fadeTime); - } else { - throw new IllegalArgumentException("Instance is not an LightActions class."); - } + invokeMethodOf(actions).fadingLightCommand(channel, command, fadeTime); } }