[pidcontroller] Use framework scheduler and remove custom Action module (#10806)
* [pidcontroller] Use framework scheduler and remove custom Action module Signed-off-by: Fabian Wolter <github@fabian-wolter.de>
This commit is contained in:
parent
6141d69da3
commit
cf13bb8275
|
@ -14,7 +14,12 @@ A PID controller can be used for closed-loop controls. For example:
|
||||||
|
|
||||||
## Modules
|
## Modules
|
||||||
|
|
||||||
The PID controller can be used in openHAB's [rule engine](https://www.openhab.org/docs/configuration/rules-dsl.html). This automation provides a trigger and an action module.
|
The PID controller can be used in openHAB's [rule engine](https://www.openhab.org/docs/configuration/rules-dsl.html).
|
||||||
|
This automation provides a trigger module ("PID controller triggers").
|
||||||
|
The return value is used to feed the Action module "Item Action" aka "send a command", which controls the actuator.
|
||||||
|
|
||||||
|
To configure a rule, you need to add a Trigger ("PID controller triggers") and an Action ("Item Action").
|
||||||
|
Select the Item you like to control in the "Item Action" and leave the command empty.
|
||||||
|
|
||||||
### Trigger
|
### Trigger
|
||||||
|
|
||||||
|
@ -32,27 +37,18 @@ This is then transferred to the action module.
|
||||||
| `kdTimeConstant` | Decimal | D-T1: [Derivative Gain Time Constant](#derivative-time-constant-d-t1-parameter) in sec. | Y |
|
| `kdTimeConstant` | Decimal | D-T1: [Derivative Gain Time Constant](#derivative-time-constant-d-t1-parameter) in sec. | Y |
|
||||||
| `commandItem` | String | Send a String "RESET" to this item to reset the I and the D part to 0. | N |
|
| `commandItem` | String | Send a String "RESET" to this item to reset the I and the D part to 0. | N |
|
||||||
| `loopTime` | Decimal | The interval the output value will be updated in milliseconds. Note: the output will also be updated when the input value or the setpoint changes. | Y |
|
| `loopTime` | Decimal | The interval the output value will be updated in milliseconds. Note: the output will also be updated when the input value or the setpoint changes. | Y |
|
||||||
|
| `pInspector` | Item | Name of the debug Item for the current P part | N |
|
||||||
|
| `iInspector` | Item | Name of the debug Item for the current I part | N |
|
||||||
|
| `dInspector` | Item | Name of the debug Item for the current D part | N |
|
||||||
|
| `eInspector` | Item | Name of the debug Item for the current regulation difference (error) | N |
|
||||||
|
|
||||||
The `loopTime` should be max a tenth of the system response.
|
The `loopTime` should be max a tenth of the system response.
|
||||||
E.g. the heating needs 10 min to heat up the room, the loop time should be max 1 min.
|
E.g. the heating needs 10 min to heat up the room, the loop time should be max 1 min.
|
||||||
Lower values won't harm, but need more calculation resources.
|
Lower values won't harm, but need more calculation resources.
|
||||||
|
|
||||||
### Action
|
|
||||||
|
|
||||||
This module writes the PID controller's output value into the `output` Item and provides debugging abilities.
|
|
||||||
|
|
||||||
| Name | Type | Description | Required |
|
|
||||||
|--------------|------|----------------------------------------------------------------------|----------|
|
|
||||||
| `output` | Item | Name of the output Item (e.g. the valve actuator 0-100%) | Y |
|
|
||||||
| `pInspector` | Item | Name of the debug Item for the current P part | N |
|
|
||||||
| `iInspector` | Item | Name of the debug Item for the current I part | N |
|
|
||||||
| `dInspector` | Item | Name of the debug Item for the current D part | N |
|
|
||||||
| `eInspector` | Item | Name of the debug Item for the current regulation difference (error) | N |
|
|
||||||
|
|
||||||
You can view the internal P, I and D parts of the controller with the inspector Items.
|
You can view the internal P, I and D parts of the controller with the inspector Items.
|
||||||
These values are useful when tuning the controller.
|
These values are useful when tuning the controller.
|
||||||
They are updated everytime the output is updated.
|
They are updated every time the output is updated.
|
||||||
|
|
||||||
## Proportional (P) Gain Parameter
|
## Proportional (P) Gain Parameter
|
||||||
|
|
||||||
|
@ -112,8 +108,8 @@ This results in quite reasonable working systems in most cases.
|
||||||
So, this will be described in the following.
|
So, this will be described in the following.
|
||||||
|
|
||||||
To be able to proceed with this method, you need to visualize the input and the output value of the PID controller over time.
|
To be able to proceed with this method, you need to visualize the input and the output value of the PID controller over time.
|
||||||
It's also good to visualize the individual P, I and D parts (these are forming the output value) via the inspector Items.
|
It's also good to visualize the individual P, I and D parts (these are forming the output value) via the inspector items.
|
||||||
The visualization can be done by the analyze function in Main UI or by adding a persistence and use Grafana for example.
|
The visualization could be done by adding a persistence and use Grafana for example.
|
||||||
|
|
||||||
After you added a [Rule](https://www.openhab.org/docs/configuration/rules-dsl.html) with above trigger and action module and configured those, proceed with the following steps:
|
After you added a [Rule](https://www.openhab.org/docs/configuration/rules-dsl.html) with above trigger and action module and configured those, proceed with the following steps:
|
||||||
|
|
||||||
|
|
|
@ -36,5 +36,5 @@ public class PIDControllerConstants {
|
||||||
public static final String I_INSPECTOR = "iInspector";
|
public static final String I_INSPECTOR = "iInspector";
|
||||||
public static final String D_INSPECTOR = "dInspector";
|
public static final String D_INSPECTOR = "dInspector";
|
||||||
public static final String E_INSPECTOR = "eInspector";
|
public static final String E_INSPECTOR = "eInspector";
|
||||||
public static final String OUTPUT = "output";
|
public static final String COMMAND = "command";
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,9 +17,7 @@ import java.util.Set;
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||||
import org.eclipse.jdt.annotation.Nullable;
|
import org.eclipse.jdt.annotation.Nullable;
|
||||||
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerActionHandler;
|
|
||||||
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerTriggerHandler;
|
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerTriggerHandler;
|
||||||
import org.openhab.core.automation.Action;
|
|
||||||
import org.openhab.core.automation.Module;
|
import org.openhab.core.automation.Module;
|
||||||
import org.openhab.core.automation.Trigger;
|
import org.openhab.core.automation.Trigger;
|
||||||
import org.openhab.core.automation.handler.BaseModuleHandlerFactory;
|
import org.openhab.core.automation.handler.BaseModuleHandlerFactory;
|
||||||
|
@ -39,8 +37,7 @@ import org.osgi.service.component.annotations.Reference;
|
||||||
@Component(service = ModuleHandlerFactory.class, configurationPid = "action.pidcontroller")
|
@Component(service = ModuleHandlerFactory.class, configurationPid = "action.pidcontroller")
|
||||||
@NonNullByDefault
|
@NonNullByDefault
|
||||||
public class PIDControllerModuleHandlerFactory extends BaseModuleHandlerFactory {
|
public class PIDControllerModuleHandlerFactory extends BaseModuleHandlerFactory {
|
||||||
private static final Collection<String> TYPES = Set.of(PIDControllerTriggerHandler.MODULE_TYPE_ID,
|
private static final Collection<String> TYPES = Set.of(PIDControllerTriggerHandler.MODULE_TYPE_ID);
|
||||||
PIDControllerActionHandler.MODULE_TYPE_ID);
|
|
||||||
private ItemRegistry itemRegistry;
|
private ItemRegistry itemRegistry;
|
||||||
private EventPublisher eventPublisher;
|
private EventPublisher eventPublisher;
|
||||||
private BundleContext bundleContext;
|
private BundleContext bundleContext;
|
||||||
|
@ -63,8 +60,6 @@ public class PIDControllerModuleHandlerFactory extends BaseModuleHandlerFactory
|
||||||
switch (module.getTypeUID()) {
|
switch (module.getTypeUID()) {
|
||||||
case PIDControllerTriggerHandler.MODULE_TYPE_ID:
|
case PIDControllerTriggerHandler.MODULE_TYPE_ID:
|
||||||
return new PIDControllerTriggerHandler((Trigger) module, itemRegistry, eventPublisher, bundleContext);
|
return new PIDControllerTriggerHandler((Trigger) module, itemRegistry, eventPublisher, bundleContext);
|
||||||
case PIDControllerActionHandler.MODULE_TYPE_ID:
|
|
||||||
return new PIDControllerActionHandler((Action) module, itemRegistry, eventPublisher);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
|
|
|
@ -1,84 +0,0 @@
|
||||||
/**
|
|
||||||
* Copyright (c) 2010-2021 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.automation.pidcontroller.internal.handler;
|
|
||||||
|
|
||||||
import static org.openhab.automation.pidcontroller.internal.PIDControllerConstants.AUTOMATION_NAME;
|
|
||||||
|
|
||||||
import java.math.BigDecimal;
|
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
|
||||||
import org.eclipse.jdt.annotation.Nullable;
|
|
||||||
import org.openhab.core.automation.Action;
|
|
||||||
import org.openhab.core.automation.handler.ActionHandler;
|
|
||||||
import org.openhab.core.automation.handler.BaseModuleHandler;
|
|
||||||
import org.openhab.core.config.core.Configuration;
|
|
||||||
import org.openhab.core.events.EventPublisher;
|
|
||||||
import org.openhab.core.items.ItemRegistry;
|
|
||||||
import org.openhab.core.items.events.ItemCommandEvent;
|
|
||||||
import org.openhab.core.items.events.ItemEventFactory;
|
|
||||||
import org.openhab.core.library.types.DecimalType;
|
|
||||||
import org.slf4j.Logger;
|
|
||||||
import org.slf4j.LoggerFactory;
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @author Hilbrand Bouwkamp - Initial Contribution
|
|
||||||
* @author Fabian Wolter - Add PID debugging items
|
|
||||||
*/
|
|
||||||
@NonNullByDefault
|
|
||||||
public class PIDControllerActionHandler extends BaseModuleHandler<Action> implements ActionHandler {
|
|
||||||
public static final String MODULE_TYPE_ID = AUTOMATION_NAME + ".action";
|
|
||||||
|
|
||||||
private final Logger logger = LoggerFactory.getLogger(PIDControllerActionHandler.class);
|
|
||||||
|
|
||||||
private ItemRegistry itemRegistry;
|
|
||||||
private EventPublisher eventPublisher;
|
|
||||||
|
|
||||||
public PIDControllerActionHandler(Action module, ItemRegistry itemRegistry, EventPublisher eventPublisher) {
|
|
||||||
super(module);
|
|
||||||
this.itemRegistry = itemRegistry;
|
|
||||||
this.eventPublisher = eventPublisher;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public @Nullable Map<String, Object> execute(Map<String, Object> context) {
|
|
||||||
final Configuration configuration = module.getConfiguration();
|
|
||||||
|
|
||||||
context.forEach((k, v) -> {
|
|
||||||
// Remove triggername from key to get raw trigger param
|
|
||||||
String itemKey = k.substring(k.lastIndexOf('.') + 1);
|
|
||||||
String itemName = (String) configuration.get(itemKey);
|
|
||||||
|
|
||||||
if (itemName == null || itemName.isBlank()) {
|
|
||||||
// try original key name (<triggername>.<trigger_param>)
|
|
||||||
itemName = (String) configuration.get(k);
|
|
||||||
if (itemName == null || itemName.isBlank()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (v instanceof BigDecimal) {
|
|
||||||
final BigDecimal command = (BigDecimal) v;
|
|
||||||
final DecimalType outputValue = new DecimalType(command);
|
|
||||||
final ItemCommandEvent itemCommandEvent = ItemEventFactory.createCommandEvent(itemName, outputValue);
|
|
||||||
|
|
||||||
eventPublisher.post(itemCommandEvent);
|
|
||||||
} else {
|
|
||||||
logger.warn(
|
|
||||||
"Command was not posted because either the configuration was not correct or a service was missing: ItemName: {}, Command: {}, eventPublisher: {}, ItemRegistry: {}",
|
|
||||||
itemName, v, eventPublisher, itemRegistry);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -15,14 +15,10 @@ package org.openhab.automation.pidcontroller.internal.handler;
|
||||||
import static org.openhab.automation.pidcontroller.internal.PIDControllerConstants.*;
|
import static org.openhab.automation.pidcontroller.internal.PIDControllerConstants.*;
|
||||||
|
|
||||||
import java.math.BigDecimal;
|
import java.math.BigDecimal;
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.concurrent.Executors;
|
|
||||||
import java.util.concurrent.ScheduledExecutorService;
|
|
||||||
import java.util.concurrent.ScheduledFuture;
|
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||||
|
@ -32,7 +28,6 @@ import org.openhab.core.automation.ModuleHandlerCallback;
|
||||||
import org.openhab.core.automation.Trigger;
|
import org.openhab.core.automation.Trigger;
|
||||||
import org.openhab.core.automation.handler.BaseTriggerModuleHandler;
|
import org.openhab.core.automation.handler.BaseTriggerModuleHandler;
|
||||||
import org.openhab.core.automation.handler.TriggerHandlerCallback;
|
import org.openhab.core.automation.handler.TriggerHandlerCallback;
|
||||||
import org.openhab.core.common.NamedThreadFactory;
|
|
||||||
import org.openhab.core.config.core.Configuration;
|
import org.openhab.core.config.core.Configuration;
|
||||||
import org.openhab.core.events.Event;
|
import org.openhab.core.events.Event;
|
||||||
import org.openhab.core.events.EventFilter;
|
import org.openhab.core.events.EventFilter;
|
||||||
|
@ -44,6 +39,7 @@ import org.openhab.core.items.ItemRegistry;
|
||||||
import org.openhab.core.items.events.ItemEventFactory;
|
import org.openhab.core.items.events.ItemEventFactory;
|
||||||
import org.openhab.core.items.events.ItemStateChangedEvent;
|
import org.openhab.core.items.events.ItemStateChangedEvent;
|
||||||
import org.openhab.core.items.events.ItemStateEvent;
|
import org.openhab.core.items.events.ItemStateEvent;
|
||||||
|
import org.openhab.core.library.types.DecimalType;
|
||||||
import org.openhab.core.library.types.StringType;
|
import org.openhab.core.library.types.StringType;
|
||||||
import org.openhab.core.types.RefreshType;
|
import org.openhab.core.types.RefreshType;
|
||||||
import org.openhab.core.types.State;
|
import org.openhab.core.types.State;
|
||||||
|
@ -63,18 +59,19 @@ public class PIDControllerTriggerHandler extends BaseTriggerModuleHandler implem
|
||||||
public static final String MODULE_TYPE_ID = AUTOMATION_NAME + ".trigger";
|
public static final String MODULE_TYPE_ID = AUTOMATION_NAME + ".trigger";
|
||||||
private static final Set<String> SUBSCRIBED_EVENT_TYPES = Set.of(ItemStateEvent.TYPE, ItemStateChangedEvent.TYPE);
|
private static final Set<String> SUBSCRIBED_EVENT_TYPES = Set.of(ItemStateEvent.TYPE, ItemStateChangedEvent.TYPE);
|
||||||
private final Logger logger = LoggerFactory.getLogger(PIDControllerTriggerHandler.class);
|
private final Logger logger = LoggerFactory.getLogger(PIDControllerTriggerHandler.class);
|
||||||
private final ScheduledExecutorService scheduler = Executors
|
|
||||||
.newSingleThreadScheduledExecutor(new NamedThreadFactory("automation-" + AUTOMATION_NAME, true));
|
|
||||||
private final ServiceRegistration<?> eventSubscriberRegistration;
|
private final ServiceRegistration<?> eventSubscriberRegistration;
|
||||||
private final PIDController controller;
|
private final PIDController controller;
|
||||||
private final int loopTimeMs;
|
private final int loopTimeMs;
|
||||||
private @Nullable ScheduledFuture<?> controllerjob;
|
|
||||||
private long previousTimeMs = System.currentTimeMillis();
|
private long previousTimeMs = System.currentTimeMillis();
|
||||||
private Item inputItem;
|
private Item inputItem;
|
||||||
private Item setpointItem;
|
private Item setpointItem;
|
||||||
private Optional<String> commandTopic;
|
private Optional<String> commandTopic;
|
||||||
private EventFilter eventFilter;
|
private EventFilter eventFilter;
|
||||||
private EventPublisher eventPublisher;
|
private EventPublisher eventPublisher;
|
||||||
|
private @Nullable String pInspector;
|
||||||
|
private @Nullable String iInspector;
|
||||||
|
private @Nullable String dInspector;
|
||||||
|
private @Nullable String eInspector;
|
||||||
|
|
||||||
public PIDControllerTriggerHandler(Trigger module, ItemRegistry itemRegistry, EventPublisher eventPublisher,
|
public PIDControllerTriggerHandler(Trigger module, ItemRegistry itemRegistry, EventPublisher eventPublisher,
|
||||||
BundleContext bundleContext) {
|
BundleContext bundleContext) {
|
||||||
|
@ -109,6 +106,10 @@ public class PIDControllerTriggerHandler extends BaseTriggerModuleHandler implem
|
||||||
double kiAdjuster = getDoubleFromConfig(config, CONFIG_KI_GAIN);
|
double kiAdjuster = getDoubleFromConfig(config, CONFIG_KI_GAIN);
|
||||||
double kdAdjuster = getDoubleFromConfig(config, CONFIG_KD_GAIN);
|
double kdAdjuster = getDoubleFromConfig(config, CONFIG_KD_GAIN);
|
||||||
double kdTimeConstant = getDoubleFromConfig(config, CONFIG_KD_TIMECONSTANT);
|
double kdTimeConstant = getDoubleFromConfig(config, CONFIG_KD_TIMECONSTANT);
|
||||||
|
pInspector = (String) config.get(P_INSPECTOR);
|
||||||
|
iInspector = (String) config.get(I_INSPECTOR);
|
||||||
|
dInspector = (String) config.get(D_INSPECTOR);
|
||||||
|
eInspector = (String) config.get(E_INSPECTOR);
|
||||||
|
|
||||||
loopTimeMs = ((BigDecimal) requireNonNull(config.get(CONFIG_LOOP_TIME), CONFIG_LOOP_TIME + " is not set"))
|
loopTimeMs = ((BigDecimal) requireNonNull(config.get(CONFIG_LOOP_TIME), CONFIG_LOOP_TIME + " is not set"))
|
||||||
.intValue();
|
.intValue();
|
||||||
|
@ -118,17 +119,21 @@ public class PIDControllerTriggerHandler extends BaseTriggerModuleHandler implem
|
||||||
eventFilter = event -> {
|
eventFilter = event -> {
|
||||||
String topic = event.getTopic();
|
String topic = event.getTopic();
|
||||||
|
|
||||||
return topic.equals("openhab/items/" + inputItemName + "/state")
|
return ("openhab/items/" + inputItemName + "/state").equals(topic)
|
||||||
|| topic.equals("openhab/items/" + inputItemName + "/statechanged")
|
|| ("openhab/items/" + inputItemName + "/statechanged").equals(topic)
|
||||||
|| topic.equals("openhab/items/" + setpointItemName + "/statechanged")
|
|| ("openhab/items/" + setpointItemName + "/statechanged").equals(topic)
|
||||||
|| commandTopic.map(t -> topic.equals(t)).orElse(false);
|
|| commandTopic.map(t -> topic.equals(t)).orElse(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
eventSubscriberRegistration = bundleContext.registerService(EventSubscriber.class.getName(), this, null);
|
eventSubscriberRegistration = bundleContext.registerService(EventSubscriber.class.getName(), this, null);
|
||||||
|
|
||||||
eventPublisher.post(ItemEventFactory.createCommandEvent(inputItemName, RefreshType.REFRESH));
|
eventPublisher.post(ItemEventFactory.createCommandEvent(inputItemName, RefreshType.REFRESH));
|
||||||
|
}
|
||||||
|
|
||||||
controllerjob = scheduler.scheduleWithFixedDelay(this::calculate, 0, loopTimeMs, TimeUnit.MILLISECONDS);
|
@Override
|
||||||
|
public void setCallback(ModuleHandlerCallback callback) {
|
||||||
|
super.setCallback(callback);
|
||||||
|
getCallback().getScheduler().scheduleWithFixedDelay(this::calculate, 0, loopTimeMs, TimeUnit.MILLISECONDS);
|
||||||
}
|
}
|
||||||
|
|
||||||
private <T> T requireNonNull(T obj, String message) {
|
private <T> T requireNonNull(T obj, String message) {
|
||||||
|
@ -165,24 +170,27 @@ public class PIDControllerTriggerHandler extends BaseTriggerModuleHandler implem
|
||||||
PIDOutputDTO output = controller.calculate(input, setpoint, now - previousTimeMs, loopTimeMs);
|
PIDOutputDTO output = controller.calculate(input, setpoint, now - previousTimeMs, loopTimeMs);
|
||||||
previousTimeMs = now;
|
previousTimeMs = now;
|
||||||
|
|
||||||
Map<String, BigDecimal> outputs = new HashMap<>();
|
updateItem(pInspector, output.getProportionalPart());
|
||||||
|
updateItem(iInspector, output.getIntegralPart());
|
||||||
|
updateItem(dInspector, output.getDerivativePart());
|
||||||
|
updateItem(eInspector, output.getError());
|
||||||
|
|
||||||
putBigDecimal(outputs, OUTPUT, output.getOutput());
|
getCallback().triggered(module, Map.of(COMMAND, new DecimalType(output.getOutput())));
|
||||||
putBigDecimal(outputs, P_INSPECTOR, output.getProportionalPart());
|
}
|
||||||
putBigDecimal(outputs, I_INSPECTOR, output.getIntegralPart());
|
|
||||||
putBigDecimal(outputs, D_INSPECTOR, output.getDerivativePart());
|
|
||||||
putBigDecimal(outputs, E_INSPECTOR, output.getError());
|
|
||||||
|
|
||||||
ModuleHandlerCallback localCallback = callback;
|
private void updateItem(@Nullable String itemName, double value) {
|
||||||
if (localCallback != null && localCallback instanceof TriggerHandlerCallback) {
|
if (itemName != null) {
|
||||||
((TriggerHandlerCallback) localCallback).triggered(module, outputs);
|
eventPublisher.post(ItemEventFactory.createCommandEvent(itemName, new DecimalType(value)));
|
||||||
} else {
|
|
||||||
logger.warn("No callback set");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void putBigDecimal(Map<String, BigDecimal> map, String key, double value) {
|
private TriggerHandlerCallback getCallback() {
|
||||||
map.put(key, BigDecimal.valueOf(value));
|
ModuleHandlerCallback localCallback = callback;
|
||||||
|
if (localCallback != null && localCallback instanceof TriggerHandlerCallback) {
|
||||||
|
return (TriggerHandlerCallback) localCallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new IllegalStateException("The module callback is not set");
|
||||||
}
|
}
|
||||||
|
|
||||||
private double getItemValueAsNumber(Item item) throws PIDException {
|
private double getItemValueAsNumber(Item item) throws PIDException {
|
||||||
|
@ -191,7 +199,7 @@ public class PIDControllerTriggerHandler extends BaseTriggerModuleHandler implem
|
||||||
if (setpointState instanceof Number) {
|
if (setpointState instanceof Number) {
|
||||||
double doubleValue = ((Number) setpointState).doubleValue();
|
double doubleValue = ((Number) setpointState).doubleValue();
|
||||||
|
|
||||||
if (Double.isFinite(doubleValue)) {
|
if (Double.isFinite(doubleValue) && !Double.isNaN(doubleValue)) {
|
||||||
return doubleValue;
|
return doubleValue;
|
||||||
}
|
}
|
||||||
} else if (setpointState instanceof StringType) {
|
} else if (setpointState instanceof StringType) {
|
||||||
|
@ -237,13 +245,6 @@ public class PIDControllerTriggerHandler extends BaseTriggerModuleHandler implem
|
||||||
public void dispose() {
|
public void dispose() {
|
||||||
eventSubscriberRegistration.unregister();
|
eventSubscriberRegistration.unregister();
|
||||||
|
|
||||||
ScheduledFuture<?> localControllerjob = controllerjob;
|
|
||||||
if (localControllerjob != null) {
|
|
||||||
localControllerjob.cancel(true);
|
|
||||||
}
|
|
||||||
|
|
||||||
scheduler.shutdown();
|
|
||||||
|
|
||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,15 +14,11 @@ package org.openhab.automation.pidcontroller.internal.template;
|
||||||
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||||
import org.openhab.automation.pidcontroller.internal.PIDControllerConstants;
|
|
||||||
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerActionHandler;
|
|
||||||
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerTriggerHandler;
|
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerTriggerHandler;
|
||||||
import org.openhab.automation.pidcontroller.internal.type.PIDControllerActionType;
|
|
||||||
import org.openhab.core.automation.Action;
|
import org.openhab.core.automation.Action;
|
||||||
import org.openhab.core.automation.Condition;
|
import org.openhab.core.automation.Condition;
|
||||||
import org.openhab.core.automation.Trigger;
|
import org.openhab.core.automation.Trigger;
|
||||||
|
@ -45,15 +41,8 @@ public class PIDControllerRuleTemplate extends RuleTemplate {
|
||||||
final List<Trigger> triggers = List.of(ModuleBuilder.createTrigger().withId(triggerId)
|
final List<Trigger> triggers = List.of(ModuleBuilder.createTrigger().withId(triggerId)
|
||||||
.withTypeUID(PIDControllerTriggerHandler.MODULE_TYPE_ID).withLabel("PID Controller Trigger").build());
|
.withTypeUID(PIDControllerTriggerHandler.MODULE_TYPE_ID).withLabel("PID Controller Trigger").build());
|
||||||
|
|
||||||
final Map<String, String> actionInputs = Map.of(PIDControllerActionType.INPUT,
|
return new PIDControllerRuleTemplate(Set.of("PID Controller"), triggers, Collections.emptyList(),
|
||||||
triggerId + "." + PIDControllerConstants.OUTPUT);
|
Collections.emptyList(), Collections.emptyList());
|
||||||
|
|
||||||
final List<Action> actions = List.of(ModuleBuilder.createAction().withId(UUID.randomUUID().toString())
|
|
||||||
.withTypeUID(PIDControllerActionHandler.MODULE_TYPE_ID).withLabel("PID Controller Action")
|
|
||||||
.withInputs(actionInputs).build());
|
|
||||||
|
|
||||||
return new PIDControllerRuleTemplate(Set.of("PID Controller"), triggers, Collections.emptyList(), actions,
|
|
||||||
Collections.emptyList());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public PIDControllerRuleTemplate(Set<String> tags, List<Trigger> triggers, List<Condition> conditions,
|
public PIDControllerRuleTemplate(Set<String> tags, List<Trigger> triggers, List<Condition> conditions,
|
||||||
|
|
|
@ -1,71 +0,0 @@
|
||||||
/**
|
|
||||||
* Copyright (c) 2010-2021 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.automation.pidcontroller.internal.type;
|
|
||||||
|
|
||||||
import static org.openhab.automation.pidcontroller.internal.PIDControllerConstants.*;
|
|
||||||
|
|
||||||
import java.math.BigDecimal;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
|
||||||
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerActionHandler;
|
|
||||||
import org.openhab.core.automation.Visibility;
|
|
||||||
import org.openhab.core.automation.type.ActionType;
|
|
||||||
import org.openhab.core.automation.type.Input;
|
|
||||||
import org.openhab.core.config.core.ConfigDescriptionParameter;
|
|
||||||
import org.openhab.core.config.core.ConfigDescriptionParameter.Type;
|
|
||||||
import org.openhab.core.config.core.ConfigDescriptionParameterBuilder;
|
|
||||||
|
|
||||||
/**
|
|
||||||
*
|
|
||||||
* @author Hilbrand Bouwkamp - Initial Contribution
|
|
||||||
*/
|
|
||||||
@NonNullByDefault
|
|
||||||
public class PIDControllerActionType extends ActionType {
|
|
||||||
public static final String INPUT = "input";
|
|
||||||
|
|
||||||
public static PIDControllerActionType initialize() {
|
|
||||||
final ConfigDescriptionParameter outputItem = ConfigDescriptionParameterBuilder.create(OUTPUT, Type.TEXT)
|
|
||||||
.withRequired(true).withMultiple(false).withContext("item").withLabel("Output Item")
|
|
||||||
.withDescription("Item to send output").build();
|
|
||||||
final ConfigDescriptionParameter pInspectorItem = ConfigDescriptionParameterBuilder
|
|
||||||
.create(P_INSPECTOR, Type.TEXT).withRequired(false).withMultiple(false).withContext("item")
|
|
||||||
.withLabel("P Inspector Item").withDescription("Item for debugging the P part").build();
|
|
||||||
final ConfigDescriptionParameter iInspectorItem = ConfigDescriptionParameterBuilder
|
|
||||||
.create(I_INSPECTOR, Type.TEXT).withRequired(false).withMultiple(false).withContext("item")
|
|
||||||
.withLabel("I Inspector Item").withDescription("Item for debugging the I part").build();
|
|
||||||
final ConfigDescriptionParameter dInspectorItem = ConfigDescriptionParameterBuilder
|
|
||||||
.create(D_INSPECTOR, Type.TEXT).withRequired(false).withMultiple(false).withContext("item")
|
|
||||||
.withLabel("D Inspector Item").withDescription("Item for debugging the D part").build();
|
|
||||||
final ConfigDescriptionParameter eInspectorItem = ConfigDescriptionParameterBuilder
|
|
||||||
.create(E_INSPECTOR, Type.TEXT).withRequired(false).withMultiple(false).withContext("item")
|
|
||||||
.withLabel("Error Inspector Item").withDescription("Item for debugging the error value").build();
|
|
||||||
|
|
||||||
List<ConfigDescriptionParameter> config = List.of(outputItem, pInspectorItem, iInspectorItem, dInspectorItem,
|
|
||||||
eInspectorItem);
|
|
||||||
|
|
||||||
List<Input> inputs = List.of(createInput(INPUT), createInput(P_INSPECTOR), createInput(I_INSPECTOR),
|
|
||||||
createInput(D_INSPECTOR), createInput(E_INSPECTOR));
|
|
||||||
|
|
||||||
return new PIDControllerActionType(config, inputs);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Input createInput(String name) {
|
|
||||||
return new Input(name, BigDecimal.class.getName());
|
|
||||||
}
|
|
||||||
|
|
||||||
public PIDControllerActionType(List<ConfigDescriptionParameter> configDescriptions, List<Input> inputs) {
|
|
||||||
super(PIDControllerActionHandler.MODULE_TYPE_ID, configDescriptions, "calculate PID output", null, null,
|
|
||||||
Visibility.VISIBLE, inputs, null);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -19,7 +19,6 @@ import java.util.Map;
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||||
import org.eclipse.jdt.annotation.Nullable;
|
import org.eclipse.jdt.annotation.Nullable;
|
||||||
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerActionHandler;
|
|
||||||
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerTriggerHandler;
|
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerTriggerHandler;
|
||||||
import org.openhab.core.automation.type.ModuleType;
|
import org.openhab.core.automation.type.ModuleType;
|
||||||
import org.openhab.core.automation.type.ModuleTypeProvider;
|
import org.openhab.core.automation.type.ModuleTypeProvider;
|
||||||
|
@ -33,9 +32,8 @@ import org.osgi.service.component.annotations.Component;
|
||||||
@Component
|
@Component
|
||||||
@NonNullByDefault
|
@NonNullByDefault
|
||||||
public class PIDControllerModuleTypeProvider implements ModuleTypeProvider {
|
public class PIDControllerModuleTypeProvider implements ModuleTypeProvider {
|
||||||
private static final Map<String, ModuleType> PROVIDED_MODULE_TYPES = Map.of(
|
private static final Map<String, ModuleType> PROVIDED_MODULE_TYPES = Map
|
||||||
PIDControllerActionHandler.MODULE_TYPE_ID, PIDControllerActionType.initialize(),
|
.of(PIDControllerTriggerHandler.MODULE_TYPE_ID, PIDControllerTriggerType.initialize());
|
||||||
PIDControllerTriggerHandler.MODULE_TYPE_ID, PIDControllerTriggerType.initialize());
|
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
@Override
|
@Override
|
||||||
|
|
|
@ -17,6 +17,7 @@ import static org.openhab.automation.pidcontroller.internal.PIDControllerConstan
|
||||||
import java.math.BigDecimal;
|
import java.math.BigDecimal;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
import org.eclipse.jdt.annotation.NonNullByDefault;
|
import org.eclipse.jdt.annotation.NonNullByDefault;
|
||||||
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerTriggerHandler;
|
import org.openhab.automation.pidcontroller.internal.handler.PIDControllerTriggerHandler;
|
||||||
|
@ -30,24 +31,26 @@ import org.openhab.core.config.core.ConfigDescriptionParameterBuilder;
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @author Hilbrand Bouwkamp - Initial Contribution
|
* @author Hilbrand Bouwkamp - Initial Contribution
|
||||||
|
* @author Fabian Wolter - Add inspector Items for debugging
|
||||||
*/
|
*/
|
||||||
@NonNullByDefault
|
@NonNullByDefault
|
||||||
public class PIDControllerTriggerType extends TriggerType {
|
public class PIDControllerTriggerType extends TriggerType {
|
||||||
private static final String DEFAULT_LOOPTIME_MS = "1000";
|
private static final String DEFAULT_LOOPTIME_MS = "1000";
|
||||||
|
private static final String ITEM = "item";
|
||||||
|
|
||||||
public static PIDControllerTriggerType initialize() {
|
public static PIDControllerTriggerType initialize() {
|
||||||
List<ConfigDescriptionParameter> configDescriptions = new ArrayList<>();
|
List<ConfigDescriptionParameter> configDescriptions = new ArrayList<>();
|
||||||
configDescriptions.add(ConfigDescriptionParameterBuilder.create(CONFIG_INPUT_ITEM, Type.TEXT) //
|
configDescriptions.add(ConfigDescriptionParameterBuilder.create(CONFIG_INPUT_ITEM, Type.TEXT) //
|
||||||
.withRequired(true) //
|
.withRequired(true) //
|
||||||
.withMultiple(false) //
|
.withMultiple(false) //
|
||||||
.withContext("item") //
|
.withContext(ITEM) //
|
||||||
.withLabel("Input Item") //
|
.withLabel("Input Item") //
|
||||||
.withDescription("Item to monitor") //
|
.withDescription("Item to monitor") //
|
||||||
.build());
|
.build());
|
||||||
configDescriptions.add(ConfigDescriptionParameterBuilder.create(CONFIG_SETPOINT_ITEM, Type.TEXT) //
|
configDescriptions.add(ConfigDescriptionParameterBuilder.create(CONFIG_SETPOINT_ITEM, Type.TEXT) //
|
||||||
.withRequired(true) //
|
.withRequired(true) //
|
||||||
.withMultiple(false) //
|
.withMultiple(false) //
|
||||||
.withContext("item") //
|
.withContext(ITEM) //
|
||||||
.withLabel("Setpoint") //
|
.withLabel("Setpoint") //
|
||||||
.withDescription("Targeted setpoint") //
|
.withDescription("Targeted setpoint") //
|
||||||
.build());
|
.build());
|
||||||
|
@ -83,13 +86,6 @@ public class PIDControllerTriggerType extends TriggerType {
|
||||||
.withDescription("Slows the rate of change of the D part (T1) in seconds.") //
|
.withDescription("Slows the rate of change of the D part (T1) in seconds.") //
|
||||||
.withUnit("s") //
|
.withUnit("s") //
|
||||||
.build());
|
.build());
|
||||||
configDescriptions.add(ConfigDescriptionParameterBuilder.create(CONFIG_COMMAND_ITEM, Type.TEXT) //
|
|
||||||
.withRequired(false) //
|
|
||||||
.withMultiple(false) //
|
|
||||||
.withContext("item") //
|
|
||||||
.withLabel("Command Item") //
|
|
||||||
.withDescription("You can send String commands to this Item like \"RESET\".") //
|
|
||||||
.build());
|
|
||||||
configDescriptions.add(ConfigDescriptionParameterBuilder.create(CONFIG_LOOP_TIME, Type.DECIMAL) //
|
configDescriptions.add(ConfigDescriptionParameterBuilder.create(CONFIG_LOOP_TIME, Type.DECIMAL) //
|
||||||
.withRequired(true) //
|
.withRequired(true) //
|
||||||
.withMultiple(false) //
|
.withMultiple(false) //
|
||||||
|
@ -98,20 +94,37 @@ public class PIDControllerTriggerType extends TriggerType {
|
||||||
.withDescription("The interval the output value is updated in ms") //
|
.withDescription("The interval the output value is updated in ms") //
|
||||||
.withUnit("ms") //
|
.withUnit("ms") //
|
||||||
.build());
|
.build());
|
||||||
Output output = new Output(OUTPUT, BigDecimal.class.getName(), "Output", "Output value of the PID Controller",
|
configDescriptions.add(ConfigDescriptionParameterBuilder.create(P_INSPECTOR, Type.TEXT) //
|
||||||
null, null, null);
|
.withRequired(false) //
|
||||||
Output pInspector = new Output(P_INSPECTOR, BigDecimal.class.getName(), "P Inspector",
|
.withMultiple(false) //
|
||||||
"Current P value of the pid controller", null, null, null);
|
.withContext(ITEM) //
|
||||||
Output iInspector = new Output(I_INSPECTOR, BigDecimal.class.getName(), "I Inspector",
|
.withLabel("P Inspector Item") //
|
||||||
"Current I value of the pid controller", null, null, null);
|
.withDescription("Item for debugging the P part") //
|
||||||
Output dInspector = new Output(D_INSPECTOR, BigDecimal.class.getName(), "D Inspector",
|
.build());
|
||||||
"Current D value of the pid controller", null, null, null);
|
configDescriptions.add(ConfigDescriptionParameterBuilder.create(I_INSPECTOR, Type.TEXT) //
|
||||||
Output eInspector = new Output(E_INSPECTOR, BigDecimal.class.getName(), "Error Value Inspector",
|
.withRequired(false) //
|
||||||
"Current error value of the pid controller", null, null, null);
|
.withMultiple(false) //
|
||||||
|
.withContext(ITEM) //
|
||||||
|
.withLabel("I Inspector Item") //
|
||||||
|
.withDescription("Item for debugging the I part") //
|
||||||
|
.build());
|
||||||
|
configDescriptions.add(ConfigDescriptionParameterBuilder.create(D_INSPECTOR, Type.TEXT) //
|
||||||
|
.withRequired(false).withMultiple(false) //
|
||||||
|
.withContext(ITEM) //
|
||||||
|
.withLabel("D Inspector Item") //
|
||||||
|
.withDescription("Item for debugging the D part") //
|
||||||
|
.build());
|
||||||
|
configDescriptions.add(ConfigDescriptionParameterBuilder.create(E_INSPECTOR, Type.TEXT) //
|
||||||
|
.withRequired(false).withMultiple(false) //
|
||||||
|
.withContext(ITEM) //
|
||||||
|
.withLabel("Error Inspector Item") //
|
||||||
|
.withDescription("Item for debugging the error value") //
|
||||||
|
.build());
|
||||||
|
|
||||||
List<Output> outputs = List.of(output, pInspector, iInspector, dInspector, eInspector);
|
Output output = new Output(COMMAND, BigDecimal.class.getName(), "Output", "Output value of the PID Controller",
|
||||||
|
Set.of("command"), null, null);
|
||||||
|
|
||||||
return new PIDControllerTriggerType(configDescriptions, outputs);
|
return new PIDControllerTriggerType(configDescriptions, List.of(output));
|
||||||
}
|
}
|
||||||
|
|
||||||
public PIDControllerTriggerType(List<ConfigDescriptionParameter> configDescriptions, List<Output> outputs) {
|
public PIDControllerTriggerType(List<ConfigDescriptionParameter> configDescriptions, List<Output> outputs) {
|
||||||
|
|
Loading…
Reference in New Issue