mf_engine/handler/function/module/
zen.rs1use std::future::Future;
2use std::pin::Pin;
3use std::sync::Arc;
4
5use crate::handler::custom_node_adapter::CustomNodeAdapter;
6use crate::handler::function::error::{FunctionResult, ResultExt};
7use crate::handler::function::listener::{RuntimeEvent, RuntimeListener};
8use crate::handler::function::module::export_default;
9use crate::handler::function::serde::JsValue;
10use crate::handler::graph::{DecisionGraph, DecisionGraphConfig};
11use crate::loader::DecisionLoader;
12use rquickjs::module::{Declarations, Exports, ModuleDef};
13use rquickjs::prelude::{Async, Func, Opt};
14use rquickjs::{CatchResultExt, Ctx, Function, Object};
15
16pub struct ZenListener<Loader, Adapter> {
17 pub loader: Arc<Loader>,
18 pub adapter: Arc<Adapter>,
19}
20
21impl<Loader: DecisionLoader + 'static, Adapter: CustomNodeAdapter + 'static>
22 RuntimeListener for ZenListener<Loader, Adapter>
23{
24 fn on_event<'js>(
25 &self,
26 ctx: Ctx<'js>,
27 event: RuntimeEvent,
28 ) -> Pin<Box<dyn Future<Output = FunctionResult> + 'js>> {
29 let loader = self.loader.clone();
30 let adapter = self.adapter.clone();
31
32 Box::pin(async move {
33 if event != RuntimeEvent::Startup {
34 return Ok(());
35 };
36
37 ctx.globals()
38 .set(
39 "__evaluate",
40 Func::from(Async(
41 move |ctx: Ctx<'js>,
42 key: String,
43 context: JsValue,
44 opts: Opt<Object<'js>>| {
45 let loader = loader.clone();
46 let adapter = adapter.clone();
47
48 async move {
49 let config: Object = ctx
50 .globals()
51 .get("config")
52 .or_throw(&ctx)?;
53
54 let iteration: u8 =
55 config.get("iteration").or_throw(&ctx)?;
56 let max_depth: u8 =
57 config.get("maxDepth").or_throw(&ctx)?;
58 let trace = opts
59 .0
60 .map(|opt| {
61 opt.get::<_, bool>("trace")
62 .unwrap_or_default()
63 })
64 .unwrap_or_default();
65
66 let load_result =
67 loader.load(key.as_str()).await;
68 let decision_content =
69 load_result.or_throw(&ctx)?;
70 let mut sub_tree = DecisionGraph::try_new(
71 DecisionGraphConfig {
72 content: decision_content,
73 max_depth,
74 loader,
75 adapter,
76 iteration: iteration + 1,
77 trace,
78 validator_cache: None,
79 },
80 )
81 .or_throw(&ctx)?;
82
83 let response = sub_tree
84 .evaluate(context.0)
85 .await
86 .or_throw(&ctx)?;
87 let k = serde_json::to_value(response)
88 .or_throw(&ctx)?
89 .into();
90
91 return rquickjs::Result::Ok(JsValue(k));
92 }
93 },
94 )),
95 )
96 .catch(&ctx)?;
97
98 Ok(())
99 })
100 }
101}
102
103fn evaluate_expression<'js>(
104 ctx: Ctx<'js>,
105 expression: String,
106 context: JsValue,
107) -> rquickjs::Result<JsValue> {
108 let s = mf_expression::evaluate_expression(expression.as_str(), context.0)
109 .or_throw(&ctx)?;
110
111 Ok(JsValue(s))
112}
113
114fn evaluate_unary_expression<'js>(
115 ctx: Ctx<'js>,
116 expression: String,
117 context: JsValue,
118) -> rquickjs::Result<bool> {
119 let s = mf_expression::evaluate_unary_expression(
120 expression.as_str(),
121 context.0,
122 )
123 .or_throw(&ctx)?;
124
125 Ok(s)
126}
127
128fn evaluate<'js>(
129 ctx: Ctx<'js>,
130 key: String,
131 context: JsValue,
132 opts: Opt<Object<'js>>,
133) -> rquickjs::Result<rquickjs::Value<'js>> {
134 let s: Function = ctx.globals().get("__evaluate").or_throw(&ctx)?;
135 let result: rquickjs::Value =
136 s.call((key, context, opts)).or_throw(&ctx)?;
137 Ok(result)
138}
139
140pub struct ZenModule;
141
142impl ModuleDef for ZenModule {
143 fn declare<'js>(decl: &Declarations<'js>) -> rquickjs::Result<()> {
144 decl.declare("evaluateExpression")?;
145 decl.declare("evaluateUnaryExpression")?;
146 decl.declare("evaluate")?;
147
148 decl.declare("default")?;
149
150 Ok(())
151 }
152
153 fn evaluate<'js>(
154 ctx: &Ctx<'js>,
155 exports: &Exports<'js>,
156 ) -> rquickjs::Result<()> {
157 export_default(ctx, exports, |default| {
158 default
159 .set("evaluateExpression", Func::from(evaluate_expression))?;
160 default.set(
161 "evaluateUnaryExpression",
162 Func::from(evaluate_unary_expression),
163 )?;
164 default.set("evaluate", Func::from(evaluate))?;
165
166 Ok(())
167 })
168 }
169}