stof 0.9.6

Data that carries its own logic.
Documentation
//
// Copyright 2025 Formata, Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//    http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//

use std::{cell::RefCell, collections::BTreeMap, sync::Arc};
use imbl::vector;
use js_sys::{Function, Promise};
use serde::{Deserialize, Serialize};
use wasm_bindgen::prelude::*;
use wasm_bindgen_futures::JsFuture;
use crate::{js::value::{to_graph_value, to_raw_value}, model::{Graph, LibFunc, stof_std::THROW}, runtime::{Error, Val, ValRef, Variable, WakeRef, instruction::{Instruction, Instructions}, instructions::Base, proc::ProcEnv, wake}};


thread_local! {
    static JS_FUNCTIONS: RefCell<BTreeMap<String, BTreeMap<String, Function>>> = RefCell::new(BTreeMap::default());
}


#[wasm_bindgen]
/// JS Library Function.
pub struct StofFunc {
    func: LibFunc,
}
impl StofFunc {
    pub fn get_func(self) -> LibFunc {
        self.func
    }

    fn set_js_func(lib: &str, name: &str, func: Function) {
        JS_FUNCTIONS.with_borrow_mut(|map| {
            if let Some(lib) = map.get_mut(lib) {
                lib.insert(name.into(), func);
            } else {
                let mut inner = BTreeMap::new();
                inner.insert(name.into(), func);
                map.insert(lib.into(), inner);
            }
        });
    }
}
#[wasm_bindgen]
impl StofFunc {
    #[wasm_bindgen(constructor)]
    /// Create a new Stof function from a JS function.
    pub fn new(library: &str, name: &str, js_function: JsValue, is_async: bool) -> Self {
        let js_function = Function::from(js_function);
        Self::set_js_func(library, name, js_function);

        let lib = library.to_string();
        let nm = name.to_string();
        let func = LibFunc {
            library: library.into(),
            name: name.into(),
            is_async,
            docs: String::default(),
            params: vector![],
            unbounded_args: true,
            return_type: None,
            args_to_symbol_table: false,
            func: Arc::new(move |_as_ref, arg_count, _env, _graph| {
                let mut instructions = Instructions::default();
                instructions.push(Arc::new(JsLibFuncIns::Call(arg_count, lib.clone(), nm.clone())));
                Ok(instructions)
            }),
        };

        Self { func }
    }
}


#[derive(Debug, Clone, Serialize, Deserialize)]
/// JS Library Function Instructions.
enum JsLibFuncIns {
    Call(usize, String, String),
}
#[typetag::serde(name = "JsLibFuncIns")]
impl Instruction for JsLibFuncIns {
    fn exec(&self, env: &mut ProcEnv, graph: &mut Graph) -> Result<Option<Instructions>, Error> {
        match self {
            Self::Call(arg_count, library, name) => {
                let context = JsValue::from(Val::Obj(env.self_ptr()));
                let res = JS_FUNCTIONS.with_borrow(|map| {
                    if let Some(lib) = map.get(library) {
                        if let Some(js_func) = lib.get(name) {
                            match arg_count {
                                0 => {
                                    js_func.call0(&context)
                                },
                                1 => {
                                    js_func.call1(&context, &env.stack.pop().unwrap().val.read().clone().into())
                                },
                                2 => {
                                    let one = env.stack.pop().unwrap().val.read().clone();
                                    let zer = env.stack.pop().unwrap().val.read().clone();
                                    js_func.call2(&context, &zer.into(), &one.into())
                                },
                                3 => {
                                    let two = env.stack.pop().unwrap().val.read().clone();
                                    let one = env.stack.pop().unwrap().val.read().clone();
                                    let zer = env.stack.pop().unwrap().val.read().clone();
                                    js_func.call3(&context, &zer.into(), &one.into(), &two.into())
                                },
                                4 => {
                                    let thr = env.stack.pop().unwrap().val.read().clone();
                                    let two = env.stack.pop().unwrap().val.read().clone();
                                    let one = env.stack.pop().unwrap().val.read().clone();
                                    let zer = env.stack.pop().unwrap().val.read().clone();
                                    js_func.call4(&context, &zer.into(), &one.into(), &two.into(), &thr.into())
                                },
                                5 => {
                                    let foy = env.stack.pop().unwrap().val.read().clone();
                                    let thr = env.stack.pop().unwrap().val.read().clone();
                                    let two = env.stack.pop().unwrap().val.read().clone();
                                    let one = env.stack.pop().unwrap().val.read().clone();
                                    let zer = env.stack.pop().unwrap().val.read().clone();
                                    js_func.call5(&context, &zer.into(), &one.into(), &two.into(), &thr.into(), &foy.into())
                                },
                                6 => {
                                    let six = env.stack.pop().unwrap().val.read().clone();
                                    let foy = env.stack.pop().unwrap().val.read().clone();
                                    let thr = env.stack.pop().unwrap().val.read().clone();
                                    let two = env.stack.pop().unwrap().val.read().clone();
                                    let one = env.stack.pop().unwrap().val.read().clone();
                                    let zer = env.stack.pop().unwrap().val.read().clone();
                                    js_func.call6(&context, &zer.into(), &one.into(), &two.into(), &thr.into(), &foy.into(), &six.into())
                                },
                                7 => {
                                    let sev = env.stack.pop().unwrap().val.read().clone();
                                    let six = env.stack.pop().unwrap().val.read().clone();
                                    let foy = env.stack.pop().unwrap().val.read().clone();
                                    let thr = env.stack.pop().unwrap().val.read().clone();
                                    let two = env.stack.pop().unwrap().val.read().clone();
                                    let one = env.stack.pop().unwrap().val.read().clone();
                                    let zer = env.stack.pop().unwrap().val.read().clone();
                                    js_func.call7(&context, &zer.into(), &one.into(), &two.into(), &thr.into(), &foy.into(), &six.into(), &sev.into())
                                },
                                8 => {
                                    let eig = env.stack.pop().unwrap().val.read().clone();
                                    let sev = env.stack.pop().unwrap().val.read().clone();
                                    let six = env.stack.pop().unwrap().val.read().clone();
                                    let foy = env.stack.pop().unwrap().val.read().clone();
                                    let thr = env.stack.pop().unwrap().val.read().clone();
                                    let two = env.stack.pop().unwrap().val.read().clone();
                                    let one = env.stack.pop().unwrap().val.read().clone();
                                    let zer = env.stack.pop().unwrap().val.read().clone();
                                    js_func.call8(&context, &zer.into(), &one.into(), &two.into(), &thr.into(), &foy.into(), &six.into(), &sev.into(), &eig.into())
                                },
                                9 => {
                                    let nin = env.stack.pop().unwrap().val.read().clone();
                                    let eig = env.stack.pop().unwrap().val.read().clone();
                                    let sev = env.stack.pop().unwrap().val.read().clone();
                                    let six = env.stack.pop().unwrap().val.read().clone();
                                    let foy = env.stack.pop().unwrap().val.read().clone();
                                    let thr = env.stack.pop().unwrap().val.read().clone();
                                    let two = env.stack.pop().unwrap().val.read().clone();
                                    let one = env.stack.pop().unwrap().val.read().clone();
                                    let zer = env.stack.pop().unwrap().val.read().clone();
                                    js_func.call9(&context, &zer.into(), &one.into(), &two.into(), &thr.into(), &foy.into(), &six.into(), &sev.into(), &eig.into(), &nin.into())
                                },
                                _ => {
                                    Err(JsValue::from_str("outnumbered allotted argument count for JS/Stof interop"))
                                }
                            }
                        } else {
                            Err(JsValue::from_str(&format!("JS/Stof Function not found: {library}.{name}")))
                        }
                    } else {
                        Err(JsValue::from_str(&format!("JS/Stof Function not found: {library}.{name}")))
                    }
                });
                match res {
                    Ok(result) => {
                        // if the result is a promise, do the async things
                        if result.is_instance_of::<Promise>() {
                            let promise = Promise::from(result);
                            
                            let wake_ref = WakeRef::default(); // when to return
                            let placeholder = ValRef::new(Val::Null); // the return value

                            // start the JS promise in the background
                            let wake_clone = wake_ref.clone();
                            let ret_val = placeholder.clone();
                            wasm_bindgen_futures::spawn_local(async move {
                                match JsFuture::from(promise).await {
                                    Ok(result) => {
                                        let mut ret = ret_val.write();
                                        *ret = to_raw_value(result);
                                    },
                                    Err(error) => {
                                        let mut ret = ret_val.write();
                                        *ret = to_raw_value(error);
                                    }
                                }
                                //web_sys::console::log_1(&"calling wake".into());
                                wake(&wake_clone); // wake the Stof process after complete
                            });

                            let mut instructions = Instructions::default();
                            instructions.push(Arc::new(Base::Variable(Variable::refval(placeholder.clone()))));
                            instructions.push(Arc::new(Base::CtrlSleepRef(wake_ref)));
                            return Ok(Some(instructions));
                        } else {
                            env.stack.push(Variable::val(to_graph_value(result, &graph)));
                        }
                    },
                    Err(error) => {
                        let mut instructions = Instructions::default();
                        instructions.push(Arc::new(Base::Literal(to_graph_value(error, &graph))));
                        instructions.push(THROW.clone());
                        return Ok(Some(instructions));
                    }
                }
            },
        }
        Ok(None)
    }
}