1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
use std::{
    io::{stdout, Write},
    process,
    str::FromStr,
};
#[derive(Parser, Debug, Clone)]
#[command(version, about, long_about = None)]
#[clap(name = "rqjs", version = env!("CARGO_PKG_VERSION"), bin_name = "rqjs")]
pub struct Args {
    #[clap()]
    file: Option<String>,
}

use clap::Parser;
use rquickjs::{
    async_with,
    loader::{BuiltinResolver, ModuleLoader},
    AsyncContext, AsyncRuntime, Ctx, Function, Module, Object, Result, Value,
};

use crate::modules;
fn print(s: String) {
    print!("{s}");
}

pub async fn start(args: Args) {
    let Args { file } = args;
    let resolver = (
        BuiltinResolver::default().with_module("os"),
        BuiltinResolver::default().with_module("path"),
        BuiltinResolver::default().with_module("buffer"),
        BuiltinResolver::default().with_module("util"),
        BuiltinResolver::default().with_module("uuid"),
        BuiltinResolver::default().with_module("xml"),
        BuiltinResolver::default().with_module("fs"),
        BuiltinResolver::default().with_module("fs/promises"),
    );

    let loader = (
        ModuleLoader::default().with_module("os", modules::os::OsModule),
        ModuleLoader::default().with_module("path", modules::path::PathModule),
        ModuleLoader::default().with_module("buffer", modules::buffer::BufferModule),
        ModuleLoader::default().with_module("util", modules::util::UtilModule),
        ModuleLoader::default().with_module("uuid", modules::uuid::UuidModule),
        ModuleLoader::default().with_module("xml", modules::xml::XmlModule),
        ModuleLoader::default().with_module("fs", modules::fs::FsModule),
        ModuleLoader::default().with_module("fs/promises", modules::fs::FsPromisesModule),
    );

    let init_global: Vec<fn(&Ctx<'_>) -> Result<()>> = vec![
        modules::buffer::init,
        modules::exceptions::init,
        modules::encoding::init,
    ];
    //     let rquickjs_version = "0.5.1";
    //     let quickjs_version = "2020-01-19";
    //     let rqjs_version = env!("CARGO_PKG_VERSION");
    //     println!(
    //         r#"Welcome to rqjs
    // rqjs: v{rqjs_version}
    // quickjs: v{quickjs_version}
    // rquickjs: v{rquickjs_version}
    // exit using ctrl+c"#
    //     );

    let core = include_str!("../deno-scripts/00_core.js");
    // let rt = Runtime::new().unwrap();
    // rt.set_loader(resolver, loader);
    // let ctx = Context::full(&rt).unwrap();
    let rt = AsyncRuntime::new().unwrap();
    rt.set_loader(resolver, loader).await;
    let ctx = AsyncContext::full(&rt).await.unwrap();

    ctx.with(|ctx| {
        let global = ctx.globals();
        global
            .set(
                "__print",
                Function::new(ctx.clone(), print)
                    .unwrap()
                    .with_name("__print")
                    .unwrap(),
            )
            .unwrap();

        let _: Value = ctx.eval(core.as_bytes()).unwrap();
    })
    .await;

    if let Some(path) = file {
        let path = std::path::PathBuf::from_str(&path).unwrap();
        let path = std::path::Path::new(&path);
        let name = path.file_name().expect("filename error");
        let code = std::fs::read_to_string(path).expect("read file error");
        async_with!(ctx => |ctx|  {
            // modules::buffer::init(&ctx).unwrap();
            for i in init_global {
                i(&ctx).unwrap();
            }

            Module::evaluate(ctx.clone(), name.to_string_lossy().to_string(), code)
                .unwrap()
                .finish::<()>()
                .unwrap();
        })
        .await;
        rt.idle().await;
    } else {
        ctrlc::set_handler(move || {
            process::exit(0);
        })
        .expect("Error setting Ctrl-C handler");

        loop {
            stdout().write_all(b"> ").unwrap();
            stdout().flush().unwrap();
            let mut input = String::new();
            std::io::stdin().read_line(&mut input).unwrap();

            ctx.with(|ctx| {
                let r = ctx.eval::<Value, _>(input.as_bytes()).unwrap();
                let g = ctx.globals();
                let console: Object = g.get("console").unwrap();
                let log: Function = console.get("log").unwrap();
                log.call::<(Value<'_>,), ()>((r,)).unwrap();
            })
            .await;
        }
    }
}