summaryrefslogtreecommitdiff
path: root/module/ri-engine/src/main
diff options
context:
space:
mode:
Diffstat (limited to 'module/ri-engine/src/main')
-rw-r--r--module/ri-engine/src/main/java/io/trygvis/rules/engine/Engine.java45
-rw-r--r--module/ri-engine/src/main/java/io/trygvis/rules/engine/JinjavaTemplateEngine.java9
-rw-r--r--module/ri-engine/src/main/java/io/trygvis/rules/engine/Main.java3
-rw-r--r--module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/EngineFile.java18
-rw-r--r--module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/NinjaCommand.java156
-rw-r--r--module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/RunCommand.java13
6 files changed, 225 insertions, 19 deletions
diff --git a/module/ri-engine/src/main/java/io/trygvis/rules/engine/Engine.java b/module/ri-engine/src/main/java/io/trygvis/rules/engine/Engine.java
index ef6b4b1..46a3302 100644
--- a/module/ri-engine/src/main/java/io/trygvis/rules/engine/Engine.java
+++ b/module/ri-engine/src/main/java/io/trygvis/rules/engine/Engine.java
@@ -1,6 +1,7 @@
package io.trygvis.rules.engine;
import org.drools.core.audit.WorkingMemoryConsoleLogger;
+import org.drools.core.base.MapGlobalResolver;
import org.drools.reflective.classloader.ProjectClassLoader;
import org.kie.api.KieServices;
import org.kie.api.event.rule.AgendaEventListener;
@@ -11,6 +12,7 @@ import org.kie.api.runtime.KieSession;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
+import javax.annotation.Nullable;
import java.io.Closeable;
import java.io.File;
import java.io.FileNotFoundException;
@@ -24,20 +26,25 @@ import java.util.List;
import static io.trygvis.rules.engine.TemplateEngine.TemplateLoader;
public class Engine implements Closeable {
+ @SuppressWarnings("FieldCanBeLocal")
private final Logger logger = LoggerFactory.getLogger(getClass());
+ public final String name;
+ @Nullable
+ public final File output;
public final DbIo io;
public final KieSession session;
- public Engine(String name, File[] databases, File output, String[] agendaGroups, File[] modules) throws IOException {
+ public Engine(String name, File[] databases, @Nullable File output, String[] agendaGroups, File[] modules)
+ throws IOException {
+ this.name = name;
+ this.output = output;
+
logger.info("Getting KieServices");
var services = KieServices.Factory.get();
- logger.info("services = {}", services);
-
var kieRepository = services.getRepository();
- logger.info("kieRepository.getDefaultReleaseId() = {}", kieRepository.getDefaultReleaseId());
KieContainer container;
TemplateLoader templateLoader;
@@ -85,12 +92,7 @@ public class Engine implements Closeable {
session.addEventListener((AgendaEventListener) l);
session.addEventListener((RuleRuntimeEventListener) l);
- var te = session.getGlobals().get("te");
- try {
- session.setGlobal("te", new JinjavaTemplateEngine(templateLoader, output));
- } catch (java.lang.RuntimeException ignore) {
- // This happens if the rules doesn't need the template engine.
- }
+ session.getGlobals().setDelegate(new EngineGlobalResolver(templateLoader));
logger.info("Loading data");
io = new DbIo(container, kieBase);
@@ -146,4 +148,27 @@ public class Engine implements Closeable {
}
}
}
+
+ private class EngineGlobalResolver extends MapGlobalResolver {
+ private final TemplateLoader templateLoader;
+
+ public EngineGlobalResolver() {
+ templateLoader = null;
+ }
+
+ public EngineGlobalResolver(TemplateLoader templateLoader) {
+ this.templateLoader = templateLoader;
+ }
+
+ @Override
+ public Object resolveGlobal(String identifier) {
+ if ("te".equals(identifier)) {
+ if (output == null) {
+ throw new IllegalArgumentException("An instance of the TemplateEngine is required, but this job is not configured with a output directory.");
+ }
+ return new JinjavaTemplateEngine(templateLoader, output);
+ }
+ return super.resolveGlobal(identifier);
+ }
+ }
}
diff --git a/module/ri-engine/src/main/java/io/trygvis/rules/engine/JinjavaTemplateEngine.java b/module/ri-engine/src/main/java/io/trygvis/rules/engine/JinjavaTemplateEngine.java
index 286029e..ba23089 100644
--- a/module/ri-engine/src/main/java/io/trygvis/rules/engine/JinjavaTemplateEngine.java
+++ b/module/ri-engine/src/main/java/io/trygvis/rules/engine/JinjavaTemplateEngine.java
@@ -3,22 +3,29 @@ package io.trygvis.rules.engine;
import ch.qos.logback.core.util.FileUtil;
import com.hubspot.jinjava.Jinjava;
import org.apache.commons.io.FileUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
import java.io.File;
import java.io.IOException;
import java.nio.file.Files;
import java.util.Map;
+import java.util.Objects;
/**
* TODO: cache templates.
*/
public class JinjavaTemplateEngine implements TemplateEngine {
+ private final Logger logger = LoggerFactory.getLogger(getClass());
+
private final Jinjava jinjava = new Jinjava();
private final TemplateLoader loader;
private final File basedir;
public JinjavaTemplateEngine(TemplateLoader templateLoader, File basedir) {
+ Objects.requireNonNull(templateLoader);
+ Objects.requireNonNull(basedir);
this.loader = templateLoader;
this.basedir = basedir;
}
@@ -26,7 +33,7 @@ public class JinjavaTemplateEngine implements TemplateEngine {
@Override
public void clean() {
try {
- System.out.println("Cleaning gen!");
+ logger.info("Removing output directory: {}", basedir);
FileUtils.deleteDirectory(basedir);
} catch (IOException e) {
throw new RuntimeException(e);
diff --git a/module/ri-engine/src/main/java/io/trygvis/rules/engine/Main.java b/module/ri-engine/src/main/java/io/trygvis/rules/engine/Main.java
index 4f06091..dc9ed02 100644
--- a/module/ri-engine/src/main/java/io/trygvis/rules/engine/Main.java
+++ b/module/ri-engine/src/main/java/io/trygvis/rules/engine/Main.java
@@ -1,13 +1,14 @@
package io.trygvis.rules.engine;
import io.trygvis.rules.engine.cli.DatabaseCommand;
+import io.trygvis.rules.engine.cli.NinjaCommand;
import io.trygvis.rules.engine.cli.RunCommand;
import picocli.CommandLine;
import picocli.CommandLine.Command;
@Command(
name = "engine",
- subcommands = {RunCommand.class, DatabaseCommand.class},
+ subcommands = {RunCommand.class, DatabaseCommand.class, NinjaCommand.class},
mixinStandardHelpOptions = true,
version = "UNSPECIFIED")
class Main {
diff --git a/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/EngineFile.java b/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/EngineFile.java
new file mode 100644
index 0000000..8da0e39
--- /dev/null
+++ b/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/EngineFile.java
@@ -0,0 +1,18 @@
+package io.trygvis.rules.engine.cli;
+
+import java.util.ArrayList;
+import java.util.List;
+
+public class EngineFile {
+ public String dbDir;
+ public List<Job> jobs;
+
+ public static class Job {
+ public String name;
+ public List<String> inputs = new ArrayList<>();
+ public List<String> outputIncludes = new ArrayList<>();
+ public String generatedOutput;
+ public List<String> agendaGroups = new ArrayList<>();
+ public List<String> modules = new ArrayList<>();
+ }
+}
diff --git a/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/NinjaCommand.java b/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/NinjaCommand.java
new file mode 100644
index 0000000..ef5ed1d
--- /dev/null
+++ b/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/NinjaCommand.java
@@ -0,0 +1,156 @@
+package io.trygvis.rules.engine.cli;
+
+import com.fasterxml.jackson.databind.MapperFeature;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.fasterxml.jackson.dataformat.yaml.YAMLFactory;
+import com.fasterxml.jackson.dataformat.yaml.YAMLGenerator;
+import org.apache.commons.lang3.StringUtils;
+
+import java.io.File;
+import java.io.FileWriter;
+import java.io.PrintWriter;
+import java.io.StringWriter;
+import java.nio.file.Path;
+import java.util.List;
+import java.util.Objects;
+import java.util.concurrent.Callable;
+import java.util.stream.Collectors;
+
+import static picocli.CommandLine.Command;
+
+@Command(name = "ninja")
+public class NinjaCommand implements Callable<Integer> {
+
+ public File basedir = null;
+
+ public Path basepath;
+
+ @Override
+ public Integer call() throws Exception {
+ basepath = Objects.requireNonNullElseGet(basedir, () -> new File("").getAbsoluteFile()).toPath();
+
+ var factory = new YAMLFactory();
+ factory.enable(YAMLGenerator.Feature.USE_NATIVE_TYPE_ID);
+ factory.enable(YAMLGenerator.Feature.USE_NATIVE_OBJECT_ID);
+ var mapper = new ObjectMapper(factory);
+ mapper.enable(MapperFeature.AUTO_DETECT_FIELDS);
+
+ var f = mapper.readValue(new File(basedir, "engine.yaml"), EngineFile.class);
+
+ Path dbDir;
+ if (StringUtils.trimToNull(f.dbDir) == null) {
+ System.err.println("Missing required field: dbDir");
+ return 1;
+ } else {
+ dbDir = Path.of(f.dbDir);
+ }
+
+ var buf = new StringWriter();
+ var out = new PrintWriter(buf);
+
+ out.println("# Generated");
+ out.println("#");
+ out.println("### engine.ninja");
+ out.println("");
+ out.println("rule engine-yaml-to-ninja");
+ out.println(" command = engine ninja");
+ out.println("");
+ out.println("build engine.ninja: engine-yaml-to-ninja engine.yaml");
+ out.println("");
+ out.println("### engine.png");
+ out.println("");
+ out.println("rule ninja-to-dot");
+ out.println(" command = ninja -t graph > $out");
+ out.println("");
+ out.println("rule dot-to-png");
+ out.println(" command = dot -Tpng < $in > $out");
+ out.println("");
+ out.println("build engine.dot: ninja-to-dot build.ninja engine.ninja");
+ out.println("");
+ out.println("build engine.png: dot-to-png engine.dot");
+ out.println("");
+ out.println("# Jobs");
+
+ for (var job : f.jobs) {
+ out.println("");
+
+ out.println("rule %s".formatted(job.name));
+ out.println(" command=engine run $name $inputs $output_state $output_includes $generated_output $agenda_groups $modules");
+ out.println();
+ var dependencies = job.inputs.stream()
+ .map(s -> dbDir.resolve(s + ".yaml").toString())
+ .collect(Collectors.joining(" "));
+
+ var outputState = dbDir.resolve(job.name + ".yaml");
+
+ var generated = List.of(outputState).stream()
+ .map(Path::toString)
+ .collect(Collectors.joining(" "));
+
+ out.println("build %s: %s %s".formatted(generated, job.name, dependencies));
+ out.println(" name=--name %s".formatted(job.name));
+
+ if (!job.inputs.isEmpty()) {
+ var is = job.inputs.stream()
+ .map(s -> "--input=" + dbDir.resolve(s + ".yaml"))
+ .collect(Collectors.joining(" $\n ", "\n ", ""));
+
+ out.println(" inputs=$%s".formatted(is));
+ }
+
+ out.println(" output_state=--output-state %s".formatted(outputState));
+ if (!job.outputIncludes.isEmpty()) {
+ var str = job.outputIncludes.stream()
+ .map(s -> "--output-include=" + s)
+ .collect(Collectors.joining(" $\n ", "\n ", ""));
+
+ out.println(" output_includes=$%s".formatted(str));
+ }
+
+ if (job.generatedOutput != null) {
+ out.println(" generated_output=--generated-output %s".formatted(fixPath(job.generatedOutput)));
+ }
+
+ if (!job.agendaGroups.isEmpty()) {
+ var ag = job.agendaGroups.stream()
+ .map(s -> "--agenda-group=" + s)
+ .collect(Collectors.joining(" $\n ", "\n ", ""));
+ out.println(" agenda_groups=%s".formatted(ag));
+ }
+
+ if (!job.modules.isEmpty()) {
+ var ms = job.modules.stream()
+ .map(this::fixPath)
+ .map(s -> "--module=" + s)
+ .collect(Collectors.joining(" $\n ", "\n ", ""));
+
+ out.println(" modules=$%s".formatted(ms));
+ }
+ }
+
+ var ninjaFile = new File(basedir, "engine.ninja");
+ try (var writer = new FileWriter(ninjaFile)) {
+ writer.write(buf.toString());
+ }
+
+ return 0;
+ }
+
+ private String fixPath(String s) {
+ if (s.startsWith("$MODULE_HOME/")) {
+ s = "$" + s;
+ }
+
+ var p = Path.of(s);
+
+ if (p.isAbsolute()) {
+ s = basepath.relativize(Path.of(s)).toString();
+ }
+
+ if (s.contains("*")) {
+ s = "$$(echo " + s + ")";
+ }
+
+ return s;
+ }
+}
diff --git a/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/RunCommand.java b/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/RunCommand.java
index 1f3e5ed..35f30cd 100644
--- a/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/RunCommand.java
+++ b/module/ri-engine/src/main/java/io/trygvis/rules/engine/cli/RunCommand.java
@@ -20,17 +20,16 @@ public class RunCommand implements Callable<Integer> {
@Option(names = {"--output-state"})
public File outputState;
- @Option(names = {"--include"}, split = ",", arity = "1..*")
- public String[] includes;
+ @Option(names = {"--output-include"}, split = ",", arity = "1..*")
+ public String[] outputIncludes;
@Option(names = {"--generated-output"})
public File generatedOutput;
- @Option(names = {"--agenda-groups"})
+ @Option(names = {"--agenda-group"})
public String[] agendaGroups;
- // TODO: Remove --modules
- @Option(names = {"--modules", "--module"}, split = ",", arity = "1..*")
+ @Option(names = {"--module"}, split = ",", arity = "1..*")
public File[] module;
@Override
@@ -43,14 +42,14 @@ public class RunCommand implements Callable<Integer> {
try (var engine = new Engine(name, input, generatedOutput, agendaGroups, module)) {
engine.io.dump(outputState, engine.session.getFactHandles(), (Object o) ->
{
- if (includes == null || includes.length == 0) {
+ if (outputIncludes == null || outputIncludes.length == 0) {
return true;
}
var name = o.getClass().getName();
var simpleName = o.getClass().getSimpleName();
- for (var i : includes) {
+ for (var i : outputIncludes) {
var ok = false;
if (i.startsWith("*")) {
i = i.substring(1);