Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit d1b1d0b

Browse files
committedJul 6, 2024
ScriptConditionHandler/ScriptActionHandler: Add support for pre-compilation of scripts
Signed-off-by: Florian Hotze <florianh_dev@icloud.com>
1 parent c0afb3d commit d1b1d0b

File tree

5 files changed

+90
-18
lines changed

5 files changed

+90
-18
lines changed
 

‎bundles/org.openhab.core.automation.module.script/src/main/java/org/openhab/core/automation/module/script/internal/handler/AbstractScriptModuleHandler.java

+53
Original file line numberDiff line numberDiff line change
@@ -18,10 +18,14 @@
1818
import java.util.Optional;
1919
import java.util.UUID;
2020

21+
import javax.script.Compilable;
22+
import javax.script.CompiledScript;
2123
import javax.script.ScriptContext;
2224
import javax.script.ScriptEngine;
25+
import javax.script.ScriptException;
2326

2427
import org.eclipse.jdt.annotation.NonNullByDefault;
28+
import org.eclipse.jdt.annotation.Nullable;
2529
import org.openhab.core.automation.Module;
2630
import org.openhab.core.automation.handler.BaseModuleHandler;
2731
import org.openhab.core.automation.module.script.ScriptEngineContainer;
@@ -35,6 +39,7 @@
3539
*
3640
* @author Kai Kreuzer - Initial contribution
3741
* @author Simon Merschjohann - Initial contribution
42+
* @author Florian Hotze - Add support for script pre-compilation
3843
*
3944
* @param <T> the type of module the concrete handler can handle
4045
*/
@@ -54,6 +59,7 @@ public abstract class AbstractScriptModuleHandler<T extends Module> extends Base
5459
private final String engineIdentifier;
5560

5661
private Optional<ScriptEngine> scriptEngine = Optional.empty();
62+
private Optional<CompiledScript> compiledScript = Optional.empty();
5763
private final String type;
5864
protected final String script;
5965

@@ -80,6 +86,31 @@ private static String getValidConfigParameter(String parameter, Configuration co
8086
}
8187
}
8288

89+
/**
90+
* Compiles the script if the script engine supports pre-compilation.
91+
*/
92+
protected void compileScriptIfSupported() throws ScriptException {
93+
if (!scriptEngineManager.isPreCompilationSupported(type)) {
94+
return;
95+
}
96+
97+
if (compiledScript.isPresent()) {
98+
return;
99+
}
100+
Optional<ScriptEngine> engine = getScriptEngine();
101+
if (engine.isPresent()) {
102+
ScriptEngine scriptEngine = engine.get();
103+
if (scriptEngine instanceof Compilable) {
104+
logger.debug("Pre-compiling script of rule with UID '{}'", ruleUID);
105+
compiledScript = Optional.ofNullable(((Compilable) scriptEngine).compile(script));
106+
} else {
107+
logger.error(
108+
"Script engine of rule with UID '{}' does not implement Compilable but claims to support pre-compilation",
109+
module.getId());
110+
}
111+
}
112+
}
113+
83114
@Override
84115
public void dispose() {
85116
scriptEngineManager.removeEngine(engineIdentifier);
@@ -169,4 +200,26 @@ protected void resetExecutionContext(ScriptEngine engine, Map<String, ?> context
169200
executionContext.removeAttribute(key, ScriptContext.ENGINE_SCOPE);
170201
}
171202
}
203+
204+
/**
205+
* Evaluates the passed script with the ScriptEngine.
206+
*
207+
* @param engine the script engine that is used
208+
* @param script the script to evaluate
209+
* @return the value returned from the execution of the script
210+
*/
211+
protected @Nullable Object eval(ScriptEngine engine, String script) {
212+
try {
213+
if (compiledScript.isPresent()) {
214+
logger.debug("Executing pre-compiled script of rule with UID '{}'", ruleUID);
215+
return compiledScript.get().eval(engine.getContext());
216+
}
217+
logger.debug("Executing script of rule with UID '{}'", ruleUID);
218+
return engine.eval(script);
219+
} catch (ScriptException e) {
220+
logger.error("Script execution of rule with UID '{}' failed: {}", ruleUID, e.getMessage(),
221+
logger.isDebugEnabled() ? e : null);
222+
return null;
223+
}
224+
}
172225
}

‎bundles/org.openhab.core.automation.module.script/src/main/java/org/openhab/core/automation/module/script/internal/handler/ScriptActionHandler.java

+8-7
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,7 @@
3131
*
3232
* @author Kai Kreuzer - Initial contribution
3333
* @author Simon Merschjohann - Initial contribution
34+
* @author Florian Hotze - Add support for script pre-compilation
3435
*/
3536
@NonNullByDefault
3637
public class ScriptActionHandler extends AbstractScriptModuleHandler<Action> implements ActionHandler {
@@ -61,6 +62,11 @@ public void dispose() {
6162
super.dispose();
6263
}
6364

65+
@Override
66+
public void compile() throws ScriptException {
67+
super.compileScriptIfSupported();
68+
}
69+
6470
@Override
6571
public @Nullable Map<String, Object> execute(final Map<String, Object> context) {
6672
Map<String, Object> resultMap = new HashMap<>();
@@ -71,13 +77,8 @@ public void dispose() {
7177

7278
getScriptEngine().ifPresent(scriptEngine -> {
7379
setExecutionContext(scriptEngine, context);
74-
try {
75-
Object result = scriptEngine.eval(script);
76-
resultMap.put("result", result);
77-
} catch (ScriptException e) {
78-
logger.error("Script execution of rule with UID '{}' failed: {}", ruleUID, e.getMessage(),
79-
logger.isDebugEnabled() ? e : null);
80-
}
80+
Object result = eval(scriptEngine, script);
81+
resultMap.put("result", result);
8182
resetExecutionContext(scriptEngine, context);
8283
});
8384

‎bundles/org.openhab.core.automation.module.script/src/main/java/org/openhab/core/automation/module/script/internal/handler/ScriptConditionHandler.java

+13-11
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@
3030
*
3131
* @author Kai Kreuzer - Initial contribution
3232
* @author Simon Merschjohann - Initial contribution
33+
* @author Florian Hotze - Add support for script pre-compilation
3334
*/
3435
@NonNullByDefault
3536
public class ScriptConditionHandler extends AbstractScriptModuleHandler<Condition> implements ConditionHandler {
@@ -42,6 +43,11 @@ public ScriptConditionHandler(Condition module, String ruleUID, ScriptEngineMana
4243
super(module, ruleUID, scriptEngineManager);
4344
}
4445

46+
@Override
47+
public void compile() throws ScriptException {
48+
super.compileScriptIfSupported();
49+
}
50+
4551
@Override
4652
public boolean isSatisfied(final Map<String, Object> context) {
4753
boolean result = false;
@@ -55,18 +61,14 @@ public boolean isSatisfied(final Map<String, Object> context) {
5561
if (engine.isPresent()) {
5662
ScriptEngine scriptEngine = engine.get();
5763
setExecutionContext(scriptEngine, context);
58-
try {
59-
Object returnVal = scriptEngine.eval(script);
60-
if (returnVal instanceof Boolean boolean1) {
61-
result = boolean1;
62-
} else {
63-
logger.error("Script of rule with UID '{}' did not return a boolean value, but '{}'", ruleUID,
64-
returnVal);
65-
}
66-
} catch (ScriptException e) {
67-
logger.error("Script execution of rule with UID '{}' failed: {}", ruleUID, e.getMessage(),
68-
logger.isDebugEnabled() ? e : null);
64+
Object returnVal = eval(scriptEngine, script);
65+
if (returnVal instanceof Boolean boolean1) {
66+
result = boolean1;
67+
} else {
68+
logger.error("Script of rule with UID '{}' did not return a boolean value, but '{}'", ruleUID,
69+
returnVal);
6970
}
71+
resetExecutionContext(scriptEngine, context);
7072
}
7173

7274
return result;

‎bundles/org.openhab.core.automation/src/main/java/org/openhab/core/automation/handler/ActionHandler.java

+8
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,14 @@
3131
*/
3232
@NonNullByDefault
3333
public interface ActionHandler extends ModuleHandler {
34+
/**
35+
* Called to compile an {@link Action} of the {@link Rule} when the rule is initialized.
36+
*
37+
* @throws Exception if the compilation fails
38+
*/
39+
default void compile() throws Exception {
40+
// Do nothing by default
41+
}
3442

3543
/**
3644
* Called to execute an {@link Action} of the {@link Rule} when it is needed.

‎bundles/org.openhab.core.automation/src/main/java/org/openhab/core/automation/handler/ConditionHandler.java

+8
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,14 @@
2929
*/
3030
@NonNullByDefault
3131
public interface ConditionHandler extends ModuleHandler {
32+
/**
33+
* Called to compile the {@link Condition} when the {@link Rule} is initialized.
34+
*
35+
* @throws Exception if the compilation fails
36+
*/
37+
default void compile() throws Exception {
38+
// Do nothing by default
39+
}
3240

3341
/**
3442
* Checks if the Condition is satisfied in the given {@code context}.

0 commit comments

Comments
 (0)
Please sign in to comment.