dino_runtime 0.1.1

A Rust runtime for Deno
Documentation
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
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
#![allow(warnings)]

use deno_core::{cppgc, Extension, JsRuntime, PollEventLoopOptions, RuntimeOptions, op2, *};
use deno_error::JsErrorBox;
use std::path;
use std::{path::Path, string, vec};
use deno_core::v8;
use base64::{Engine as _, engine::general_purpose,alphabet};
use tokio::time::{sleep, Duration};
use tokio::net::TcpListener;
use std::borrow::Cow;
use std::cell::RefCell;
use std::rc::Rc;
use tokio::io::AsyncReadExt;
use tokio::io::AsyncWriteExt;
use tokio::net::TcpStream;
use tokio::sync::Mutex; // 非同期Mutexを使用



#[derive(serde::Serialize, serde::Deserialize)]
struct TextEncoder;
unsafe impl GarbageCollected for TextEncoder {
    fn trace(&self, _tracer: &mut v8::cppgc::Visitor) {}
    fn get_name(&self) -> &'static std::ffi::CStr {
        std::ffi::CStr::from_bytes_with_nul(b"TextEncoder\0").unwrap()
    }
}

#[op2]
fn text_encoder_new<'s, 'i>(
    scope: &mut v8::PinScope<'s, 'i>,
) -> v8::Local<'s, v8::Object> {
    cppgc::make_cppgc_object(scope, TextEncoder {})
}

#[op2]
#[buffer]
fn text_encoder_encode(
    #[cppgc] _encoder: &TextEncoder,
    #[string] input: String,
) -> Vec<u8> {
    //println!("TextEncoder encoding input: {}", input);
    input.into_bytes()
}

extension!(
    text_encoder_extension,
    ops = [
        text_encoder_new,
        text_encoder_encode,
    ],
    esm_entry_point = "internal:text_encoder_bootstrap",
    esm = [
        dir "src/js",
        "internal:text_encoder_bootstrap" = "TextEncoder.js",
    ],
);

#[derive(serde::Serialize, serde::Deserialize)]
struct TextDecoder;
unsafe impl GarbageCollected for TextDecoder {
    fn trace(&self, _tracer: &mut v8::cppgc::Visitor) {}
    fn get_name(&self) -> &'static std::ffi::CStr {
        std::ffi::CStr::from_bytes_with_nul(b"TextDecoder\0").unwrap()
    }
}

#[op2]
fn text_decoder_new<'s, 'i>(
    scope: &mut v8::PinScope<'s, 'i>,
) -> v8::Local<'s, v8::Object> {
    cppgc::make_cppgc_object(scope, TextDecoder {})
}

#[op2]
#[string]
fn text_decoder_decode(
    #[cppgc] _decoder: &TextDecoder,
    #[buffer] input: &[u8],
) -> String {
    //println!("TextDecoder decoding input: {}", input);
    String::from_utf8(input.to_vec()).unwrap()
}

extension!(
    text_decoder_extension,
    ops = [
        text_decoder_new,
        text_decoder_decode,
    ],
    esm_entry_point = "internal:text_decoder_bootstrap",
    esm = [
        dir "src/js",
        "internal:text_decoder_bootstrap" = "TextDecoder.js",
    ],
);

#[op2]
#[string]
fn to_base64(
    #[buffer] input: &[u8],
    #[string] alphabet_type: String,
    omit_padding: bool,
) -> String {
    // 1. アルファベットの選択
    let alphabet = if alphabet_type == "base64url" {
        alphabet::URL_SAFE
    } else {
        alphabet::STANDARD
    };

    // 2. パディング設定の構築
    let config = if omit_padding {
        general_purpose::NO_PAD
    } else {
        general_purpose::PAD
    };

    // 3. エンジンの作成と実行
    let engine = general_purpose::GeneralPurpose::new(&alphabet, config);
    engine.encode(input)
}

#[op2]
#[buffer]
fn from_base64(
    #[string] input: String,
    #[string] alphabet_type: String,
    omit_padding: bool,
) -> Result<Vec<u8>, JsErrorBox> {
    if !input.is_ascii() {
        return Err(JsErrorBox::generic("Input must be ASCII Base64"));
    }

    let alphabet = if alphabet_type == "base64url" {
        alphabet::URL_SAFE
    } else {
        alphabet::STANDARD
    };

    println!("alphabet: {:?}", alphabet_type);

     // 2. パディング設定の構築

    let engine = general_purpose::GeneralPurpose::new(&alphabet, general_purpose::PAD);

    match engine.decode(input.trim()) {
        Ok(bytes) => Ok(bytes),
        Err(err) =>{
            println!("Base64 decode error: {}", err);
            return Err(JsErrorBox::generic("Invalid Base64 input"));
        }
    }
}
extension!(
    base64_ext,
    ops = [to_base64, from_base64],
    esm_entry_point = "internal:base64_bootstrap",
    esm = [
        dir "src/js",
        "internal:base64_bootstrap" = "Base64.js",
    ],
);

#[op2(async(deferred), fast)]
pub async fn op_sleep_interval(
    ms: u32,
) {
    sleep(Duration::from_millis(ms as u64)).await;
}

#[op2(async(deferred), fast)]
pub async fn op_sleep_timeout(
    ms: u32,
) {
    sleep(Duration::from_millis(ms as u64)).await;
}

struct TcpListenerResource(TcpListener);
impl Resource for TcpListenerResource {
    fn name(&self) -> Cow<str> {
        "tcpListener".into()
    }
}


extension!(
    setinterval_ext,
    ops = [op_sleep_interval],
    esm_entry_point = "internal:setinterval_bootstrap",
    esm = [
        dir "src/js",
        "internal:setinterval_bootstrap" = "SetInterval.js",
    ],
);

extension!(
    settimeout_ext,
    ops = [op_sleep_timeout],
    esm_entry_point = "internal:settimeout_bootstrap",
    esm = [
        dir "src/js",
        "internal:settimeout_bootstrap" = "SetTimeout.js",
    ],
);

#[op2(fast)]
#[smi] // 数値を返すための最適化
pub fn op_net_listen(
    state: &mut OpState,
    #[string] addr: String,
) -> Result<ResourceId, JsErrorBox> {
    // 同期的にバインド(実際には非同期ライブラリですが、リソース作成までは即時)
    let std_listener = std::net::TcpListener::bind(addr)
        .map_err(|e| JsErrorBox::generic(format!("failed to bind: {}", e)))?;
    std_listener.set_nonblocking(true)
        .map_err(|e| JsErrorBox::generic(format!("failed to set non-blocking: {}", e)))?;
    let listener = TcpListener::from_std(std_listener).map_err(|e| JsErrorBox::generic(format!("failed to create TcpListener: {}", e)))?;
    
    let rid = state.resource_table.add(TcpListenerResource(listener));
    Ok(rid)
}

extension!(
    event_emitter_ext,
    esm_entry_point = "internal:event_emitter_bootstrap",
    esm = [
        dir "src/js",
        "internal:event_emitter_bootstrap" = "EventEmitter.js",
    ],
);

// これが「TcpStreamResource」の正体です
struct TcpStreamResource {
    // 読み取り用と書き込み用を分けて保持
    rd: tokio::sync::Mutex<tokio::net::tcp::OwnedReadHalf>,
    wr: tokio::sync::Mutex<tokio::net::tcp::OwnedWriteHalf>,
}

impl Resource for TcpStreamResource {
    fn name(&self) -> Cow<str> {
        "tcpStream".into()
    }
}

// 2. 接続を待機する非同期Op
#[op2(async(deferred))]
#[smi]
pub async fn op_net_accept(
    state: Rc<RefCell<OpState>>,
    rid: ResourceId,
) -> Result<ResourceId, JsErrorBox> {
    let listener = state.borrow().resource_table.get::<TcpListenerResource>(rid).map_err(|e| JsErrorBox::generic(format!("failed to get TcpListenerResource: {}", e)))?;
    
    // 接続が来るまで非同期で待機
    let (stream, _) = listener.0.accept().await
        .map_err(|e| JsErrorBox::generic(e.to_string()))?;

    let (rd, wr) = stream.into_split();
    
    // 新しい接続(TcpStream)もリソースとして登録
    let rid = state.borrow_mut().resource_table.add(TcpStreamResource {
        rd: tokio::sync::Mutex::new(rd),
        wr: tokio::sync::Mutex::new(wr),
    });
    Ok(rid)
}

#[op2(fast)]
pub fn op_net_close(
    state: &mut OpState,
    #[smi] rid: ResourceId,
) -> Result<(), JsErrorBox> {
    // ResourceTableから取り除くことで、Rust側のオブジェクトがDropされる
    // これによりOSレベルでソケットが閉じられる
    state.resource_table.close(rid)
        .map_err(|_| JsErrorBox::generic(format!("Resource ID {} not found", rid)))?;
    
    Ok(())
}

// 読み込み Op: 指定されたバッファにデータを読み込む
#[op2(async(deferred))]
#[smi]
pub async fn op_net_read(
    state: Rc<RefCell<OpState>>,
    #[smi] rid: ResourceId,
    #[buffer] mut buf: JsBuffer, // JS側のUint8Arrayを直接参照
) -> Result<u32, JsErrorBox> {
    // リソーステーブルからストリームを取得
    let resource = state
        .borrow()
        .resource_table
        .get::<TcpStreamResource>(rid)
        .map_err(|e| JsErrorBox::generic(e.to_string()))?;

    // state の借用が解けた状態でロックを待つ
    let mut stream = resource.rd.lock().await;

    // 3. 読み込み実行
    use tokio::io::AsyncReadExt;
    let nread = stream.read(&mut buf).await
        .map_err(|e| JsErrorBox::generic(e.to_string()))?;

    Ok(nread as u32)
}

// 書き込み Op: JSから届いたバッファをソケットへ流す
#[op2(async(deferred))]
#[smi]
pub async fn op_net_write(
    state: Rc<RefCell<OpState>>,
    #[smi] rid: ResourceId,
    #[buffer] buf: JsBuffer,
) -> Result<u32, JsErrorBox> {
    let resource = state
        .borrow()
        .resource_table
        .get::<TcpStreamResource>(rid)
        .map_err(|e| JsErrorBox::generic(e.to_string()))?;
    let mut stream = resource.wr.lock().await;
    
    // 全データが書き込まれるまで待機
    stream
        .write_all(buf.as_ref())
        .await
        .map_err(|e| JsErrorBox::generic(e.to_string()))?;
        
    Ok(buf.len() as u32)
}

#[op2]
#[serde] // 構造体をJSONとしてJSに返す
pub fn op_net_server_address(
    state: &mut OpState,
    #[smi] rid: ResourceId,
) -> Result<serde_json::Value, JsErrorBox> {
    let listener = state.resource_table.get::<TcpListenerResource>(rid)
        .map_err(|e| JsErrorBox::generic(e.to_string()))?;
    let addr = listener.0.local_addr()
        .map_err(|e| JsErrorBox::generic(e.to_string()))?;

    Ok(serde_json::json!({
        "address": addr.ip().to_string(),
        "port": addr.port(),
        "family": if addr.is_ipv4() { "IPv4" } else { "IPv6" }
    }))
}

#[op2(async(deferred), fast)]
#[smi]
pub async fn op_net_connect(
    state: Rc<RefCell<OpState>>,
    #[string] addr: String,
) -> Result<ResourceId, JsErrorBox> {
    // 1. 指定したアドレスに接続を試みる
    let stream = tokio::net::TcpStream::connect(addr).await
        .map_err(|e| JsErrorBox::generic(format!("Connect error: {}", e)))?;

    // 2. 読み書きを分離(デッドロック防止)
    let (rd, wr) = stream.into_split();

    // 3. リソーステーブルに登録
    let rid = state.borrow_mut().resource_table.add(TcpStreamResource {
        rd: tokio::sync::Mutex::new(rd),
        wr: tokio::sync::Mutex::new(wr),
    });

    Ok(rid)
}


extension!(
    net_ext,
    deps = [event_emitter_ext],
    ops = [op_net_listen, op_net_accept, op_net_close, op_net_read, op_net_write, op_net_server_address, op_net_connect],
    esm_entry_point = "internal:net_bootstrap",
    esm = [
        dir "src/js",
        "internal:net_bootstrap" = "Net.js",
    ],
);

pub fn run_js(path_name: String, additional_extensions: Option<Vec<Extension>>) {
    let rt = tokio::runtime::Builder::new_current_thread()
        .enable_all()
        .build()
        .unwrap();

    let path_name_static: &'static str = Box::leak(path_name.into_boxed_str()); 
    let path = Path::new(path_name_static);
    let code: &'static str  = Box::leak(std::fs::read_to_string(path).unwrap().into_boxed_str());

    let add_extensions = additional_extensions.unwrap_or_default();

    let mut extentions: Vec<Extension> = vec![
        text_encoder_extension::init(),
        text_decoder_extension::init(),
        base64_ext::init(),
        event_emitter_ext::init(),
        net_ext::init(),
    ];
    
    extentions.extend(add_extensions);

    

    rt.block_on(async move {
        let text_encoder_ext = text_encoder_extension::init();
        let text_decoder_ext = text_decoder_extension::init();
        let base64_ext = base64_ext::init();
        let event_emitter_ext = event_emitter_ext::init();
        let net_ext = net_ext::init();

        let mut runtime = JsRuntime::new(RuntimeOptions {
            extensions:extentions,
            ..Default::default()
        });

        

        runtime.execute_script(path.to_str().unwrap(), code).unwrap();
        runtime
            .run_event_loop(PollEventLoopOptions::default())
            .await
            .unwrap();
        println!("Hello, world!");
    });
}