Full game
This commit is contained in:
2
.gitignore
vendored
2
.gitignore
vendored
@@ -3,4 +3,4 @@ node_modules
|
|||||||
*.ignore
|
*.ignore
|
||||||
target
|
target
|
||||||
.DS_Store
|
.DS_Store
|
||||||
webui/src/lib/wasm
|
webui/src/wasm
|
||||||
|
|||||||
@@ -87,4 +87,4 @@ for crate in "${wasm_crates[@]}"; do
|
|||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
echo -e " • $crate → $output_dir"
|
echo -e " • $crate → $output_dir"
|
||||||
done
|
done
|
||||||
|
|||||||
37
rust/Cargo.lock
generated
37
rust/Cargo.lock
generated
@@ -339,6 +339,25 @@ dependencies = [
|
|||||||
"syn",
|
"syn",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
[[package]]
|
||||||
|
name = "runner"
|
||||||
|
version = "0.1.0"
|
||||||
|
dependencies = [
|
||||||
|
"anyhow",
|
||||||
|
"console_error_panic_hook",
|
||||||
|
"getrandom",
|
||||||
|
"itertools",
|
||||||
|
"js-sys",
|
||||||
|
"minimax",
|
||||||
|
"rand",
|
||||||
|
"rhai",
|
||||||
|
"serde",
|
||||||
|
"serde-wasm-bindgen",
|
||||||
|
"wasm-bindgen",
|
||||||
|
"web-sys",
|
||||||
|
"wee_alloc",
|
||||||
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "rustversion"
|
name = "rustversion"
|
||||||
version = "1.0.22"
|
version = "1.0.22"
|
||||||
@@ -357,24 +376,6 @@ version = "1.2.0"
|
|||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49"
|
checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49"
|
||||||
|
|
||||||
[[package]]
|
|
||||||
name = "script-runner"
|
|
||||||
version = "0.1.0"
|
|
||||||
dependencies = [
|
|
||||||
"anyhow",
|
|
||||||
"console_error_panic_hook",
|
|
||||||
"getrandom",
|
|
||||||
"js-sys",
|
|
||||||
"minimax",
|
|
||||||
"rand",
|
|
||||||
"rhai",
|
|
||||||
"serde",
|
|
||||||
"serde-wasm-bindgen",
|
|
||||||
"wasm-bindgen",
|
|
||||||
"web-sys",
|
|
||||||
"wee_alloc",
|
|
||||||
]
|
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "serde"
|
name = "serde"
|
||||||
version = "1.0.228"
|
version = "1.0.228"
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
[workspace]
|
[workspace]
|
||||||
members = ["script-runner", "minimax", "rhai-codemirror"]
|
members = ["runner", "minimax", "rhai-codemirror"]
|
||||||
resolver = "2"
|
resolver = "2"
|
||||||
|
|
||||||
[workspace.dependencies]
|
[workspace.dependencies]
|
||||||
|
|||||||
@@ -12,11 +12,15 @@ codegen-units = 1
|
|||||||
opt-level = 's'
|
opt-level = 's'
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
rhai = { workspace = true, features = ["internals", "wasm-bindgen"] }
|
rhai = { workspace = true, features = ["internals"] }
|
||||||
|
|
||||||
wasm-bindgen = { workspace = true }
|
wasm-bindgen = { workspace = true }
|
||||||
js-sys = { workspace = true }
|
js-sys = { workspace = true }
|
||||||
web-sys = { workspace = true, features = ["console"] }
|
web-sys = { workspace = true, features = ["console"] }
|
||||||
console_error_panic_hook = { workspace = true }
|
console_error_panic_hook = { workspace = true }
|
||||||
wee_alloc = { workspace = true }
|
wee_alloc = { workspace = true }
|
||||||
serde = { workspace = true }
|
serde = { workspace = true }
|
||||||
serde-wasm-bindgen = { workspace = true }
|
serde-wasm-bindgen = { workspace = true }
|
||||||
|
|
||||||
|
[target.'cfg(target_arch = "wasm32")'.dependencies]
|
||||||
|
rhai = { workspace = true, features = ["wasm-bindgen"] }
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "script-runner"
|
name = "runner"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
@@ -9,9 +9,9 @@ crate-type = ["cdylib"]
|
|||||||
[dependencies]
|
[dependencies]
|
||||||
minimax = { workspace = true }
|
minimax = { workspace = true }
|
||||||
|
|
||||||
|
|
||||||
anyhow = { workspace = true }
|
anyhow = { workspace = true }
|
||||||
rand = { workspace = true }
|
rand = { workspace = true }
|
||||||
|
itertools = { workspace = true }
|
||||||
|
|
||||||
wasm-bindgen = { workspace = true }
|
wasm-bindgen = { workspace = true }
|
||||||
js-sys = { workspace = true }
|
js-sys = { workspace = true }
|
||||||
9
rust/runner/src/ansi.rs
Normal file
9
rust/runner/src/ansi.rs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
pub const RESET: &str = "\x1b[0m";
|
||||||
|
pub const RED: &str = "\x1b[31m";
|
||||||
|
pub const BLUE: &str = "\x1b[34m";
|
||||||
|
pub const MAGENTA: &str = "\x1b[35m";
|
||||||
|
|
||||||
|
pub const UP: &str = "\x1b[A";
|
||||||
|
pub const DOWN: &str = "\x1b[B";
|
||||||
|
pub const LEFT: &str = "\x1b[D";
|
||||||
|
pub const RIGHT: &str = "\x1b[C";
|
||||||
184
rust/runner/src/human.rs
Normal file
184
rust/runner/src/human.rs
Normal file
@@ -0,0 +1,184 @@
|
|||||||
|
use itertools::Itertools;
|
||||||
|
use minimax::{
|
||||||
|
board::{Board, PlayerAction},
|
||||||
|
util::Symb,
|
||||||
|
};
|
||||||
|
|
||||||
|
use crate::ansi::{DOWN, LEFT, RESET, RIGHT, UP};
|
||||||
|
|
||||||
|
struct SymbolSelector {
|
||||||
|
symbols: Vec<char>,
|
||||||
|
cursor: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SymbolSelector {
|
||||||
|
fn new(symbols: Vec<char>) -> Self {
|
||||||
|
Self { symbols, cursor: 0 }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn current(&self) -> char {
|
||||||
|
self.symbols[self.cursor]
|
||||||
|
}
|
||||||
|
|
||||||
|
fn check(&mut self, board: &Board) {
|
||||||
|
while board.contains(Symb::from_char(self.current()).unwrap()) {
|
||||||
|
if self.cursor == 0 {
|
||||||
|
self.cursor = self.symbols.len() - 1;
|
||||||
|
} else {
|
||||||
|
self.cursor -= 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn down(&mut self, board: &Board) {
|
||||||
|
if self.cursor == 0 {
|
||||||
|
self.cursor = self.symbols.len() - 1;
|
||||||
|
} else {
|
||||||
|
self.cursor -= 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
while board.contains(Symb::from_char(self.current()).unwrap()) {
|
||||||
|
if self.cursor == 0 {
|
||||||
|
self.cursor = self.symbols.len() - 1;
|
||||||
|
} else {
|
||||||
|
self.cursor -= 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn up(&mut self, board: &Board) {
|
||||||
|
if self.cursor == self.symbols.len() - 1 {
|
||||||
|
self.cursor = 0;
|
||||||
|
} else {
|
||||||
|
self.cursor += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
while board.contains(Symb::from_char(self.current()).unwrap()) {
|
||||||
|
if self.cursor == self.symbols.len() - 1 {
|
||||||
|
self.cursor = 0;
|
||||||
|
} else {
|
||||||
|
self.cursor += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct HumanInput {
|
||||||
|
player_color: String,
|
||||||
|
cursor: usize,
|
||||||
|
symbol_selector: SymbolSelector,
|
||||||
|
|
||||||
|
/// Set to Some() when the player selects an action.
|
||||||
|
/// Should be cleared and applied immediately.
|
||||||
|
queued_action: Option<PlayerAction>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl HumanInput {
|
||||||
|
pub fn new(player_color: String) -> Self {
|
||||||
|
Self {
|
||||||
|
cursor: 0,
|
||||||
|
queued_action: None,
|
||||||
|
|
||||||
|
player_color,
|
||||||
|
symbol_selector: SymbolSelector::new(vec![
|
||||||
|
'1', '2', '3', '4', '5', '6', '7', '8', '9', '0', '+', '-', '×', '÷',
|
||||||
|
]),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn pop_action(&mut self) -> Option<PlayerAction> {
|
||||||
|
self.queued_action.take()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn print_state(&mut self, board: &Board, minimize: bool) -> String {
|
||||||
|
let cursor_max = board.size() - 1;
|
||||||
|
self.symbol_selector.check(board);
|
||||||
|
|
||||||
|
let board_label = format!(
|
||||||
|
"{}{:<6}{RESET}",
|
||||||
|
self.player_color,
|
||||||
|
if minimize { "Min" } else { "Max" },
|
||||||
|
);
|
||||||
|
|
||||||
|
return format!(
|
||||||
|
"\r{}╙{}{}{}{RESET}{}╜ {}",
|
||||||
|
board_label,
|
||||||
|
// Cursor
|
||||||
|
" ".repeat(self.cursor),
|
||||||
|
self.player_color,
|
||||||
|
if board.is_full() {
|
||||||
|
' '
|
||||||
|
} else {
|
||||||
|
self.symbol_selector.current()
|
||||||
|
},
|
||||||
|
// RESET
|
||||||
|
" ".repeat(cursor_max - self.cursor),
|
||||||
|
self.symbol_selector
|
||||||
|
.symbols
|
||||||
|
.iter()
|
||||||
|
.map(|x| {
|
||||||
|
if board.contains(Symb::from_char(*x).unwrap()) {
|
||||||
|
" ".to_string()
|
||||||
|
} else if *x == self.symbol_selector.current() {
|
||||||
|
format!("{}{x}{RESET}", self.player_color,)
|
||||||
|
} else {
|
||||||
|
format!("{x}",)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.join("")
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn process_input(&mut self, board: &Board, data: String) {
|
||||||
|
let cursor_max = board.size() - 1;
|
||||||
|
self.symbol_selector.check(board);
|
||||||
|
|
||||||
|
match &data[..] {
|
||||||
|
RIGHT => {
|
||||||
|
self.cursor = cursor_max.min(self.cursor + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
LEFT => {
|
||||||
|
if self.cursor != 0 {
|
||||||
|
self.cursor -= 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
UP => {
|
||||||
|
self.symbol_selector.up(board);
|
||||||
|
}
|
||||||
|
|
||||||
|
DOWN => {
|
||||||
|
self.symbol_selector.down(board);
|
||||||
|
}
|
||||||
|
|
||||||
|
" " | "\n" => {
|
||||||
|
let symb = Symb::from_char(self.symbol_selector.current());
|
||||||
|
if let Some(symb) = symb {
|
||||||
|
let action = PlayerAction {
|
||||||
|
symb,
|
||||||
|
pos: self.cursor,
|
||||||
|
};
|
||||||
|
|
||||||
|
if board.can_play(&action) {
|
||||||
|
self.queued_action = Some(action);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
c => {
|
||||||
|
let symb = Symb::from_str(c);
|
||||||
|
if let Some(symb) = symb {
|
||||||
|
let action = PlayerAction {
|
||||||
|
symb,
|
||||||
|
pos: self.cursor,
|
||||||
|
};
|
||||||
|
|
||||||
|
if board.can_play(&action) {
|
||||||
|
self.queued_action = Some(action);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
652
rust/runner/src/lib.rs
Normal file
652
rust/runner/src/lib.rs
Normal file
@@ -0,0 +1,652 @@
|
|||||||
|
use minimax::{
|
||||||
|
agents::{Agent, Rhai},
|
||||||
|
board::Board,
|
||||||
|
};
|
||||||
|
use rand::{rngs::StdRng, SeedableRng};
|
||||||
|
use rhai::ParseError;
|
||||||
|
use wasm_bindgen::prelude::*;
|
||||||
|
|
||||||
|
use crate::human::HumanInput;
|
||||||
|
|
||||||
|
mod ansi;
|
||||||
|
mod human;
|
||||||
|
|
||||||
|
#[global_allocator]
|
||||||
|
static ALLOC: wee_alloc::WeeAlloc = wee_alloc::WeeAlloc::INIT;
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn init_panic_hook() {
|
||||||
|
console_error_panic_hook::set_once();
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// MARK: GameState
|
||||||
|
//
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub struct GameState {
|
||||||
|
max_agent: Rhai<StdRng>,
|
||||||
|
max_name: String,
|
||||||
|
|
||||||
|
min_agent: Rhai<StdRng>,
|
||||||
|
min_name: String,
|
||||||
|
|
||||||
|
board: Board,
|
||||||
|
max_turn: bool,
|
||||||
|
|
||||||
|
game_state_callback: Box<dyn Fn(&str) + 'static>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
impl GameState {
|
||||||
|
#[wasm_bindgen(constructor)]
|
||||||
|
pub fn new(
|
||||||
|
max_script: &str,
|
||||||
|
max_name: &str,
|
||||||
|
max_print_callback: js_sys::Function,
|
||||||
|
max_debug_callback: js_sys::Function,
|
||||||
|
|
||||||
|
min_script: &str,
|
||||||
|
min_name: &str,
|
||||||
|
min_print_callback: js_sys::Function,
|
||||||
|
min_debug_callback: js_sys::Function,
|
||||||
|
|
||||||
|
game_state_callback: js_sys::Function,
|
||||||
|
) -> Result<GameState, String> {
|
||||||
|
Self::new_native(
|
||||||
|
max_script,
|
||||||
|
max_name,
|
||||||
|
move |s| {
|
||||||
|
let _ = max_print_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
||||||
|
},
|
||||||
|
move |s| {
|
||||||
|
let _ = max_debug_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
||||||
|
},
|
||||||
|
min_script,
|
||||||
|
min_name,
|
||||||
|
move |s| {
|
||||||
|
let _ = min_print_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
||||||
|
},
|
||||||
|
move |s| {
|
||||||
|
let _ = min_debug_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
||||||
|
},
|
||||||
|
move |s| {
|
||||||
|
let _ = game_state_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.map_err(|x| format!("Error at {}: {}", x.1, x.0))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn new_native(
|
||||||
|
max_script: &str,
|
||||||
|
max_name: &str,
|
||||||
|
max_print_callback: impl Fn(&str) + 'static,
|
||||||
|
max_debug_callback: impl Fn(&str) + 'static,
|
||||||
|
|
||||||
|
min_script: &str,
|
||||||
|
min_name: &str,
|
||||||
|
min_print_callback: impl Fn(&str) + 'static,
|
||||||
|
min_debug_callback: impl Fn(&str) + 'static,
|
||||||
|
|
||||||
|
game_state_callback: impl Fn(&str) + 'static,
|
||||||
|
) -> Result<GameState, ParseError> {
|
||||||
|
console_error_panic_hook::set_once();
|
||||||
|
|
||||||
|
let mut seed1 = [0u8; 32];
|
||||||
|
let mut seed2 = [0u8; 32];
|
||||||
|
getrandom::getrandom(&mut seed1).unwrap();
|
||||||
|
getrandom::getrandom(&mut seed2).unwrap();
|
||||||
|
|
||||||
|
Ok(GameState {
|
||||||
|
board: Board::new(),
|
||||||
|
max_turn: true,
|
||||||
|
|
||||||
|
max_name: max_name.to_owned(),
|
||||||
|
max_agent: Rhai::new(
|
||||||
|
max_script,
|
||||||
|
StdRng::from_seed(seed1),
|
||||||
|
max_print_callback,
|
||||||
|
max_debug_callback,
|
||||||
|
)?,
|
||||||
|
|
||||||
|
min_name: min_name.to_owned(),
|
||||||
|
min_agent: Rhai::new(
|
||||||
|
min_script,
|
||||||
|
StdRng::from_seed(seed2),
|
||||||
|
min_print_callback,
|
||||||
|
min_debug_callback,
|
||||||
|
)?,
|
||||||
|
|
||||||
|
game_state_callback: Box::new(game_state_callback),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn is_done(&self) -> bool {
|
||||||
|
self.board.is_full()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If true, it is the max player's turn.
|
||||||
|
/// If false, it is the min player's turn.
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn is_max_turn(&self) -> bool {
|
||||||
|
self.max_turn
|
||||||
|
}
|
||||||
|
|
||||||
|
fn format_board_display(&self) -> String {
|
||||||
|
let mut result = String::new();
|
||||||
|
|
||||||
|
// Board label with player name in gray
|
||||||
|
let current_player = if self.max_turn {
|
||||||
|
format!("{:<6}", self.max_name)
|
||||||
|
} else {
|
||||||
|
format!("{:<6}", self.min_name)
|
||||||
|
};
|
||||||
|
|
||||||
|
let colored_player = if self.max_turn {
|
||||||
|
format!("{}{}{}", ansi::RED, current_player, ansi::RESET)
|
||||||
|
} else {
|
||||||
|
format!("{}{}{}", ansi::BLUE, current_player, ansi::RESET)
|
||||||
|
};
|
||||||
|
result.push_str(&format!("\r{}║", colored_player));
|
||||||
|
|
||||||
|
for (i, symbol) in self.board.get_board().iter().enumerate() {
|
||||||
|
match symbol {
|
||||||
|
Some(s) => {
|
||||||
|
// Highlight the last placed symbol in magenta, everything else normal
|
||||||
|
let symbol_str = s.to_string();
|
||||||
|
if Some(i) == self.board.get_last_placed() {
|
||||||
|
result.push_str(&format!("{}{}{}", ansi::MAGENTA, symbol_str, ansi::RESET));
|
||||||
|
} else {
|
||||||
|
result.push_str(&symbol_str);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None => result.push('_'),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
result.push('║');
|
||||||
|
|
||||||
|
result
|
||||||
|
}
|
||||||
|
|
||||||
|
// Play one turn
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn step(&mut self) -> Result<Option<String>, String> {
|
||||||
|
if self.is_done() {
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
let action = match self.is_max_turn() {
|
||||||
|
true => self.max_agent.step_max(&self.board),
|
||||||
|
false => self.min_agent.step_min(&self.board),
|
||||||
|
};
|
||||||
|
|
||||||
|
let action = match action {
|
||||||
|
Ok(x) => x,
|
||||||
|
Err(err) => {
|
||||||
|
return Err(format!("{err:?}"));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if !self.board.play(
|
||||||
|
action,
|
||||||
|
self.max_turn
|
||||||
|
.then_some(&self.max_name)
|
||||||
|
.unwrap_or(&self.min_name)
|
||||||
|
.to_owned(),
|
||||||
|
) {
|
||||||
|
let error_msg = format!(
|
||||||
|
"{} {} ({}) made an invalid move {}!",
|
||||||
|
format!("{}ERROR:{}", ansi::RED, ansi::RESET),
|
||||||
|
self.max_turn
|
||||||
|
.then_some(&self.max_name)
|
||||||
|
.unwrap_or(&self.min_name),
|
||||||
|
self.max_turn
|
||||||
|
.then_some(self.max_agent.name())
|
||||||
|
.unwrap_or(self.min_agent.name()),
|
||||||
|
action
|
||||||
|
);
|
||||||
|
|
||||||
|
// Print error to game state callback
|
||||||
|
(self.game_state_callback)(&error_msg);
|
||||||
|
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
self.max_turn = !self.max_turn;
|
||||||
|
|
||||||
|
// Print board state after move to terminal (via game state callback)
|
||||||
|
let board_display = self.format_board_display();
|
||||||
|
(self.game_state_callback)(&board_display);
|
||||||
|
|
||||||
|
// Show final score if game is complete
|
||||||
|
if self.is_done() {
|
||||||
|
if let Some(score) = self.board.evaluate() {
|
||||||
|
let score_msg = format!("\nFinal score: {:.2}", score);
|
||||||
|
(self.game_state_callback)(&score_msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return Ok(Some(
|
||||||
|
self.max_turn
|
||||||
|
.then_some(&self.min_name)
|
||||||
|
.unwrap_or(&self.max_name)
|
||||||
|
.to_owned(),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// MARK: GameStateHuman
|
||||||
|
//
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub struct GameStateHuman {
|
||||||
|
human: HumanInput,
|
||||||
|
|
||||||
|
/// If true, human goes first and maximizes score.
|
||||||
|
/// If false, human goes second and minimizes score.
|
||||||
|
human_is_maximizer: bool,
|
||||||
|
|
||||||
|
agent: Rhai<StdRng>,
|
||||||
|
agent_name: String,
|
||||||
|
|
||||||
|
board: Board,
|
||||||
|
is_human_turn: bool,
|
||||||
|
is_first_turn: bool,
|
||||||
|
is_error: bool,
|
||||||
|
red_score: Option<f32>,
|
||||||
|
|
||||||
|
game_state_callback: Box<dyn Fn(&str) + 'static>,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
impl GameStateHuman {
|
||||||
|
#[wasm_bindgen(constructor)]
|
||||||
|
pub fn new(
|
||||||
|
human_is_maximizer: bool,
|
||||||
|
max_script: &str,
|
||||||
|
max_name: &str,
|
||||||
|
max_print_callback: js_sys::Function,
|
||||||
|
max_debug_callback: js_sys::Function,
|
||||||
|
|
||||||
|
game_state_callback: js_sys::Function,
|
||||||
|
) -> Result<GameStateHuman, String> {
|
||||||
|
Self::new_native(
|
||||||
|
human_is_maximizer,
|
||||||
|
max_script,
|
||||||
|
max_name,
|
||||||
|
move |s| {
|
||||||
|
let _ = max_print_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
||||||
|
},
|
||||||
|
move |s| {
|
||||||
|
let _ = max_debug_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
||||||
|
},
|
||||||
|
move |s| {
|
||||||
|
let _ = game_state_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
||||||
|
},
|
||||||
|
)
|
||||||
|
.map_err(|x| format!("Error at {}: {}", x.1, x.0))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn new_native(
|
||||||
|
human_is_maximizer: bool,
|
||||||
|
max_script: &str,
|
||||||
|
max_name: &str,
|
||||||
|
max_print_callback: impl Fn(&str) + 'static,
|
||||||
|
max_debug_callback: impl Fn(&str) + 'static,
|
||||||
|
|
||||||
|
game_state_callback: impl Fn(&str) + 'static,
|
||||||
|
) -> Result<Self, ParseError> {
|
||||||
|
console_error_panic_hook::set_once();
|
||||||
|
|
||||||
|
let mut seed1 = [0u8; 32];
|
||||||
|
let mut seed2 = [0u8; 32];
|
||||||
|
getrandom::getrandom(&mut seed1).unwrap();
|
||||||
|
getrandom::getrandom(&mut seed2).unwrap();
|
||||||
|
|
||||||
|
Ok(Self {
|
||||||
|
human_is_maximizer,
|
||||||
|
board: Board::new(),
|
||||||
|
is_human_turn: human_is_maximizer,
|
||||||
|
is_first_turn: true,
|
||||||
|
is_error: false,
|
||||||
|
red_score: None,
|
||||||
|
|
||||||
|
human: HumanInput::new(ansi::RED.to_string()),
|
||||||
|
agent_name: max_name.to_owned(),
|
||||||
|
agent: Rhai::new(
|
||||||
|
max_script,
|
||||||
|
StdRng::from_seed(seed1),
|
||||||
|
max_print_callback,
|
||||||
|
max_debug_callback,
|
||||||
|
)?,
|
||||||
|
|
||||||
|
game_state_callback: Box::new(game_state_callback),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn is_done(&self) -> bool {
|
||||||
|
(self.board.is_full() && self.red_score.is_some()) || self.is_error
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn is_error(&self) -> bool {
|
||||||
|
self.is_error
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn human_is_maximizer(&self) -> bool {
|
||||||
|
self.human_is_maximizer
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn print_start(&mut self) {
|
||||||
|
self.print_board("", "");
|
||||||
|
(self.game_state_callback)("\r\n");
|
||||||
|
|
||||||
|
if !self.is_human_turn {
|
||||||
|
let action = {
|
||||||
|
if self.human_is_maximizer {
|
||||||
|
self.agent.step_min(&self.board)
|
||||||
|
} else {
|
||||||
|
self.agent.step_max(&self.board)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let action = match action {
|
||||||
|
Ok(x) => x,
|
||||||
|
Err(err) => {
|
||||||
|
let error_msg = format!(
|
||||||
|
"{}ERROR:{} Error while computing next move: {:?}",
|
||||||
|
ansi::RED,
|
||||||
|
ansi::RESET,
|
||||||
|
err
|
||||||
|
);
|
||||||
|
|
||||||
|
(self.game_state_callback)(&error_msg);
|
||||||
|
self.is_error = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if !self.board.play(action, "Blue") {
|
||||||
|
let error_msg = format!(
|
||||||
|
"{}ERROR:{} {} made an invalid move {}!",
|
||||||
|
ansi::RED,
|
||||||
|
ansi::RESET,
|
||||||
|
self.agent_name,
|
||||||
|
action
|
||||||
|
);
|
||||||
|
|
||||||
|
(self.game_state_callback)(&error_msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
self.print_board(
|
||||||
|
self.is_human_turn
|
||||||
|
.then_some(ansi::RED)
|
||||||
|
.unwrap_or(ansi::BLUE),
|
||||||
|
self.is_human_turn.then_some("Red").unwrap_or("Blue"),
|
||||||
|
);
|
||||||
|
(self.game_state_callback)("\r\n");
|
||||||
|
self.is_human_turn = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
self.print_ui();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn print_board(&mut self, color: &str, player: &str) {
|
||||||
|
let board_label = format!("{}{:<6}{}", color, player, ansi::RESET);
|
||||||
|
|
||||||
|
// Print board
|
||||||
|
(self.game_state_callback)(&format!(
|
||||||
|
"\r{}{}{}{}",
|
||||||
|
board_label,
|
||||||
|
if self.is_first_turn { '╓' } else { '║' },
|
||||||
|
self.board.prettyprint(),
|
||||||
|
if self.is_first_turn { '╖' } else { '║' },
|
||||||
|
));
|
||||||
|
self.is_first_turn = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn print_ui(&mut self) {
|
||||||
|
(self.game_state_callback)(
|
||||||
|
&self
|
||||||
|
.human
|
||||||
|
.print_state(&self.board, !self.human_is_maximizer),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[wasm_bindgen]
|
||||||
|
pub fn take_input(&mut self, data: String) {
|
||||||
|
self.human.process_input(&self.board, data);
|
||||||
|
self.print_ui();
|
||||||
|
|
||||||
|
if let Some(action) = self.human.pop_action() {
|
||||||
|
if !self.board.play(action, "Red") {
|
||||||
|
let error_msg = format!(
|
||||||
|
"{}ERROR:{} {} made an invalid move {}!",
|
||||||
|
ansi::RED,
|
||||||
|
ansi::RESET,
|
||||||
|
"Human",
|
||||||
|
action
|
||||||
|
);
|
||||||
|
|
||||||
|
(self.game_state_callback)(&error_msg);
|
||||||
|
}
|
||||||
|
self.is_human_turn = false;
|
||||||
|
|
||||||
|
if self.board.is_full() {
|
||||||
|
self.print_end();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.print_board(ansi::RED, "Red");
|
||||||
|
(self.game_state_callback)("\r\n");
|
||||||
|
|
||||||
|
let action = {
|
||||||
|
if self.human_is_maximizer {
|
||||||
|
self.agent.step_min(&self.board)
|
||||||
|
} else {
|
||||||
|
self.agent.step_max(&self.board)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let action = match action {
|
||||||
|
Ok(x) => x,
|
||||||
|
Err(err) => {
|
||||||
|
let error_msg = format!(
|
||||||
|
"{}ERROR:{} Error while computing next move: {:?}",
|
||||||
|
ansi::RED,
|
||||||
|
ansi::RESET,
|
||||||
|
err
|
||||||
|
);
|
||||||
|
|
||||||
|
(self.game_state_callback)(&error_msg);
|
||||||
|
self.is_error = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
if !self.board.play(action, "Blue") {
|
||||||
|
let error_msg = format!(
|
||||||
|
"{}ERROR:{} {} made an invalid move {}!",
|
||||||
|
ansi::RED,
|
||||||
|
ansi::RESET,
|
||||||
|
self.agent_name,
|
||||||
|
action
|
||||||
|
);
|
||||||
|
|
||||||
|
(self.game_state_callback)(&error_msg);
|
||||||
|
}
|
||||||
|
self.is_human_turn = true;
|
||||||
|
|
||||||
|
if self.board.is_full() {
|
||||||
|
self.print_end();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.print_board(ansi::BLUE, "Blue");
|
||||||
|
(self.game_state_callback)("\r\n");
|
||||||
|
|
||||||
|
self.print_ui();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn print_end(&mut self) {
|
||||||
|
let board_label = format!(
|
||||||
|
"{}{:<6}{}",
|
||||||
|
self.is_human_turn
|
||||||
|
.then_some(ansi::BLUE)
|
||||||
|
.unwrap_or(ansi::RED),
|
||||||
|
self.is_human_turn.then_some("Blue").unwrap_or("Red"),
|
||||||
|
ansi::RESET
|
||||||
|
);
|
||||||
|
|
||||||
|
(self.game_state_callback)(&format!("\r{}║{}║", board_label, self.board.prettyprint()));
|
||||||
|
(self.game_state_callback)("\r\n");
|
||||||
|
|
||||||
|
(self.game_state_callback)(&format!(
|
||||||
|
"\r{}╙{}╜",
|
||||||
|
" ",
|
||||||
|
" ".repeat(self.board.size())
|
||||||
|
));
|
||||||
|
(self.game_state_callback)("\r\n");
|
||||||
|
|
||||||
|
let score = self.board.evaluate().unwrap();
|
||||||
|
(self.game_state_callback)(&format!(
|
||||||
|
"\r\n{}{} score:{} {:.2}\r\n",
|
||||||
|
self.human_is_maximizer
|
||||||
|
.then_some(ansi::RED)
|
||||||
|
.unwrap_or(ansi::BLUE),
|
||||||
|
self.human_is_maximizer.then_some("Red").unwrap_or("Blue"),
|
||||||
|
ansi::RESET,
|
||||||
|
score
|
||||||
|
));
|
||||||
|
(self.game_state_callback)("\r\n");
|
||||||
|
|
||||||
|
match self.red_score {
|
||||||
|
// Start second round
|
||||||
|
None => {
|
||||||
|
let mut seed1 = [0u8; 32];
|
||||||
|
let mut seed2 = [0u8; 32];
|
||||||
|
getrandom::getrandom(&mut seed1).unwrap();
|
||||||
|
getrandom::getrandom(&mut seed2).unwrap();
|
||||||
|
|
||||||
|
self.human_is_maximizer = !self.human_is_maximizer;
|
||||||
|
self.board = Board::new();
|
||||||
|
self.is_human_turn = !self.human_is_maximizer;
|
||||||
|
self.is_first_turn = true;
|
||||||
|
self.is_error = false;
|
||||||
|
self.human = HumanInput::new(ansi::RED.to_string());
|
||||||
|
self.red_score = Some(score);
|
||||||
|
|
||||||
|
self.print_start();
|
||||||
|
}
|
||||||
|
|
||||||
|
// End game
|
||||||
|
Some(red_score) => {
|
||||||
|
if red_score == score {
|
||||||
|
(self.game_state_callback)(&format!("Tie! Score: {:.2}\r\n", score));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let red_wins = red_score > score;
|
||||||
|
(self.game_state_callback)(&format!(
|
||||||
|
"{}{} wins!{}",
|
||||||
|
red_wins.then_some(ansi::RED).unwrap_or(ansi::BLUE),
|
||||||
|
red_wins.then_some("Red").unwrap_or("Blue"),
|
||||||
|
ansi::RESET,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// MARK: tests
|
||||||
|
//
|
||||||
|
// TODO:
|
||||||
|
// - infinite loop
|
||||||
|
// - random is different
|
||||||
|
// - incorrect return type
|
||||||
|
// - globals
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn full_random() {
|
||||||
|
const SCRIPT: &str = r#"
|
||||||
|
fn random_action(board) {
|
||||||
|
let symb = rand_symb();
|
||||||
|
let pos = rand_int(0, 10);
|
||||||
|
let action = Action(symb, pos);
|
||||||
|
|
||||||
|
while !board.can_play(action) {
|
||||||
|
let symb = rand_symb();
|
||||||
|
let pos = rand_int(0, 10);
|
||||||
|
action = Action(symb, pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
return action
|
||||||
|
}
|
||||||
|
|
||||||
|
fn step_min(board) {
|
||||||
|
random_action(board)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn step_max(board) {
|
||||||
|
random_action(board)
|
||||||
|
}
|
||||||
|
"#;
|
||||||
|
|
||||||
|
let mut game = GameState::new_native(
|
||||||
|
&SCRIPT,
|
||||||
|
"max",
|
||||||
|
|_| {},
|
||||||
|
|_| {},
|
||||||
|
&SCRIPT,
|
||||||
|
"min",
|
||||||
|
|_| {},
|
||||||
|
|_| {},
|
||||||
|
|_| {},
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut n = 0;
|
||||||
|
while !game.is_done() {
|
||||||
|
println!("{:?}", game.step());
|
||||||
|
println!("{:?}", game.board);
|
||||||
|
|
||||||
|
n += 1;
|
||||||
|
assert!(n < 10);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn infinite_loop() {
|
||||||
|
const SCRIPT: &str = r#"
|
||||||
|
fn step_min(board) {
|
||||||
|
loop {}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn step_max(board) {
|
||||||
|
loop {}
|
||||||
|
}
|
||||||
|
"#;
|
||||||
|
|
||||||
|
let mut game = GameState::new_native(
|
||||||
|
&SCRIPT,
|
||||||
|
"max",
|
||||||
|
|_| {},
|
||||||
|
|_| {},
|
||||||
|
&SCRIPT,
|
||||||
|
"min",
|
||||||
|
|_| {},
|
||||||
|
|_| {},
|
||||||
|
|_| {},
|
||||||
|
);
|
||||||
|
|
||||||
|
while !game.is_done() {
|
||||||
|
println!("{:?}", game.step());
|
||||||
|
println!("{:?}", game.board);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,54 +0,0 @@
|
|||||||
# Script Runner WASM
|
|
||||||
|
|
||||||
A placeholder Rust script runner using wasm-bindgen.
|
|
||||||
|
|
||||||
## Building
|
|
||||||
|
|
||||||
```bash
|
|
||||||
# Install wasm-pack if not already installed
|
|
||||||
cargo install wasm-pack
|
|
||||||
|
|
||||||
# Build for web
|
|
||||||
npm run build
|
|
||||||
|
|
||||||
# Build for different targets
|
|
||||||
npm run build:nodejs # For Node.js
|
|
||||||
npm run build:bundler # For bundlers like webpack
|
|
||||||
npm run dev # Development build
|
|
||||||
```
|
|
||||||
|
|
||||||
## Usage
|
|
||||||
|
|
||||||
```javascript
|
|
||||||
import init, { ScriptRunner, greet, add } from './pkg/script_runner.js';
|
|
||||||
|
|
||||||
async function run() {
|
|
||||||
await init();
|
|
||||||
|
|
||||||
// Basic functions
|
|
||||||
greet("World");
|
|
||||||
console.log(add(1, 2)); // 3
|
|
||||||
|
|
||||||
// Script runner
|
|
||||||
const runner = new ScriptRunner();
|
|
||||||
runner.set_context("my_context");
|
|
||||||
console.log(runner.get_context()); // "my_context"
|
|
||||||
|
|
||||||
const result = runner.run_script("console.log('Hello from script')");
|
|
||||||
console.log(result);
|
|
||||||
|
|
||||||
// Evaluate expressions
|
|
||||||
console.log(runner.evaluate("hello")); // "Hello from Rust!"
|
|
||||||
console.log(runner.evaluate("version")); // "0.1.0"
|
|
||||||
}
|
|
||||||
|
|
||||||
run();
|
|
||||||
```
|
|
||||||
|
|
||||||
## Features
|
|
||||||
|
|
||||||
- Basic WASM initialization with panic hooks
|
|
||||||
- ScriptRunner class for managing execution context
|
|
||||||
- Simple expression evaluation
|
|
||||||
- Console logging integration
|
|
||||||
- Multiple build targets (web, nodejs, bundler)
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
pub const RESET: &str = "\x1b[0m";
|
|
||||||
pub const RED: &str = "\x1b[31m";
|
|
||||||
pub const BLUE: &str = "\x1b[34m";
|
|
||||||
pub const MAGENTA: &str = "\x1b[35m";
|
|
||||||
@@ -1,317 +0,0 @@
|
|||||||
use minimax::{
|
|
||||||
agents::{Agent, Rhai},
|
|
||||||
board::Board,
|
|
||||||
};
|
|
||||||
use rand::{rngs::StdRng, SeedableRng};
|
|
||||||
use wasm_bindgen::prelude::*;
|
|
||||||
|
|
||||||
mod colors;
|
|
||||||
|
|
||||||
#[global_allocator]
|
|
||||||
static ALLOC: wee_alloc::WeeAlloc = wee_alloc::WeeAlloc::INIT;
|
|
||||||
|
|
||||||
#[wasm_bindgen]
|
|
||||||
pub fn init_panic_hook() {
|
|
||||||
console_error_panic_hook::set_once();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[wasm_bindgen]
|
|
||||||
pub struct GameState {
|
|
||||||
max_agent: Rhai<StdRng>,
|
|
||||||
max_name: String,
|
|
||||||
|
|
||||||
min_agent: Rhai<StdRng>,
|
|
||||||
min_name: String,
|
|
||||||
|
|
||||||
board: Board,
|
|
||||||
max_turn: bool,
|
|
||||||
|
|
||||||
game_state_callback: Box<dyn Fn(&str) + 'static>,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[wasm_bindgen]
|
|
||||||
impl GameState {
|
|
||||||
#[wasm_bindgen(constructor)]
|
|
||||||
pub fn new(
|
|
||||||
max_script: &str,
|
|
||||||
max_name: &str,
|
|
||||||
max_print_callback: js_sys::Function,
|
|
||||||
max_debug_callback: js_sys::Function,
|
|
||||||
|
|
||||||
min_script: &str,
|
|
||||||
min_name: &str,
|
|
||||||
min_print_callback: js_sys::Function,
|
|
||||||
min_debug_callback: js_sys::Function,
|
|
||||||
|
|
||||||
game_state_callback: js_sys::Function,
|
|
||||||
) -> GameState {
|
|
||||||
Self::new_native(
|
|
||||||
max_script,
|
|
||||||
max_name,
|
|
||||||
move |s| {
|
|
||||||
let _ = max_print_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
|
||||||
},
|
|
||||||
move |s| {
|
|
||||||
let _ = max_debug_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
|
||||||
},
|
|
||||||
min_script,
|
|
||||||
min_name,
|
|
||||||
move |s| {
|
|
||||||
let _ = min_print_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
|
||||||
},
|
|
||||||
move |s| {
|
|
||||||
let _ = min_debug_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
|
||||||
},
|
|
||||||
move |s| {
|
|
||||||
let _ = game_state_callback.call1(&JsValue::null(), &JsValue::from_str(s));
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn new_native(
|
|
||||||
max_script: &str,
|
|
||||||
max_name: &str,
|
|
||||||
max_print_callback: impl Fn(&str) + 'static,
|
|
||||||
max_debug_callback: impl Fn(&str) + 'static,
|
|
||||||
|
|
||||||
min_script: &str,
|
|
||||||
min_name: &str,
|
|
||||||
min_print_callback: impl Fn(&str) + 'static,
|
|
||||||
min_debug_callback: impl Fn(&str) + 'static,
|
|
||||||
|
|
||||||
game_state_callback: impl Fn(&str) + 'static,
|
|
||||||
) -> GameState {
|
|
||||||
console_error_panic_hook::set_once();
|
|
||||||
|
|
||||||
let mut seed1 = [0u8; 32];
|
|
||||||
let mut seed2 = [0u8; 32];
|
|
||||||
getrandom::getrandom(&mut seed1).unwrap();
|
|
||||||
getrandom::getrandom(&mut seed2).unwrap();
|
|
||||||
|
|
||||||
GameState {
|
|
||||||
board: Board::new(),
|
|
||||||
max_turn: true,
|
|
||||||
|
|
||||||
max_name: max_name.to_owned(),
|
|
||||||
max_agent: Rhai::new(
|
|
||||||
max_script,
|
|
||||||
StdRng::from_seed(seed1),
|
|
||||||
max_print_callback,
|
|
||||||
max_debug_callback,
|
|
||||||
),
|
|
||||||
|
|
||||||
min_name: min_name.to_owned(),
|
|
||||||
min_agent: Rhai::new(
|
|
||||||
min_script,
|
|
||||||
StdRng::from_seed(seed2),
|
|
||||||
min_print_callback,
|
|
||||||
min_debug_callback,
|
|
||||||
),
|
|
||||||
|
|
||||||
game_state_callback: Box::new(game_state_callback),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[wasm_bindgen]
|
|
||||||
pub fn is_done(&self) -> bool {
|
|
||||||
self.board.is_full()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// If true, it is the max player's turn.
|
|
||||||
/// If false, it is the min player's turn.
|
|
||||||
#[wasm_bindgen]
|
|
||||||
pub fn is_max_turn(&self) -> bool {
|
|
||||||
self.max_turn
|
|
||||||
}
|
|
||||||
|
|
||||||
fn format_board_display(&self) -> String {
|
|
||||||
let mut result = String::new();
|
|
||||||
|
|
||||||
// Board label with player name in gray
|
|
||||||
let current_player = if self.max_turn {
|
|
||||||
format!("{:<6}", self.max_name)
|
|
||||||
} else {
|
|
||||||
format!("{:<6}", self.min_name)
|
|
||||||
};
|
|
||||||
|
|
||||||
let colored_player = if self.max_turn {
|
|
||||||
format!("{}{}{}", colors::RED, current_player, colors::RESET)
|
|
||||||
} else {
|
|
||||||
format!("{}{}{}", colors::BLUE, current_player, colors::RESET)
|
|
||||||
};
|
|
||||||
result.push_str(&format!("\r{}║", colored_player));
|
|
||||||
|
|
||||||
for (i, symbol) in self.board.get_board().iter().enumerate() {
|
|
||||||
match symbol {
|
|
||||||
Some(s) => {
|
|
||||||
// Highlight the last placed symbol in magenta, everything else normal
|
|
||||||
let symbol_str = s.to_string();
|
|
||||||
if Some(i) == self.board.get_last_placed() {
|
|
||||||
result.push_str(&format!(
|
|
||||||
"{}{}{}",
|
|
||||||
colors::MAGENTA,
|
|
||||||
symbol_str,
|
|
||||||
colors::RESET
|
|
||||||
));
|
|
||||||
} else {
|
|
||||||
result.push_str(&symbol_str);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
None => result.push('_'),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
result.push('║');
|
|
||||||
|
|
||||||
result
|
|
||||||
}
|
|
||||||
|
|
||||||
// Play one turn
|
|
||||||
#[wasm_bindgen]
|
|
||||||
pub fn step(&mut self) -> Result<Option<String>, String> {
|
|
||||||
if self.is_done() {
|
|
||||||
return Ok(None);
|
|
||||||
}
|
|
||||||
|
|
||||||
let action = match self.is_max_turn() {
|
|
||||||
true => self.max_agent.step_max(&self.board),
|
|
||||||
false => self.min_agent.step_min(&self.board),
|
|
||||||
};
|
|
||||||
|
|
||||||
let action = match action {
|
|
||||||
Ok(x) => x,
|
|
||||||
Err(err) => {
|
|
||||||
return Err(format!("{err:?}"));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if !self.board.play(
|
|
||||||
action,
|
|
||||||
self.max_turn
|
|
||||||
.then_some(&self.max_name)
|
|
||||||
.unwrap_or(&self.min_name)
|
|
||||||
.to_owned(),
|
|
||||||
) {
|
|
||||||
let error_msg = format!(
|
|
||||||
"{} {} ({}) made an invalid move {}!",
|
|
||||||
format!("{}ERROR:{}", colors::RED, colors::RESET),
|
|
||||||
self.max_turn
|
|
||||||
.then_some(&self.max_name)
|
|
||||||
.unwrap_or(&self.min_name),
|
|
||||||
self.max_turn
|
|
||||||
.then_some(self.max_agent.name())
|
|
||||||
.unwrap_or(self.min_agent.name()),
|
|
||||||
action
|
|
||||||
);
|
|
||||||
|
|
||||||
// Print error to game state callback
|
|
||||||
(self.game_state_callback)(&error_msg);
|
|
||||||
|
|
||||||
return Ok(None);
|
|
||||||
}
|
|
||||||
|
|
||||||
self.max_turn = !self.max_turn;
|
|
||||||
|
|
||||||
// Print board state after move to terminal (via game state callback)
|
|
||||||
let board_display = self.format_board_display();
|
|
||||||
(self.game_state_callback)(&board_display);
|
|
||||||
|
|
||||||
// Show final score if game is complete
|
|
||||||
if self.is_done() {
|
|
||||||
if let Some(score) = self.board.evaluate() {
|
|
||||||
let score_msg = format!("\nFinal score: {:.2}", score);
|
|
||||||
(self.game_state_callback)(&score_msg);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Ok(Some(
|
|
||||||
self.max_turn
|
|
||||||
.then_some(&self.min_name)
|
|
||||||
.unwrap_or(&self.max_name)
|
|
||||||
.to_owned(),
|
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// TODO: tests
|
|
||||||
// - infinite loop
|
|
||||||
// - random is different
|
|
||||||
// - incorrect return type
|
|
||||||
// - globals
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn full_random() {
|
|
||||||
const SCRIPT: &str = r#"
|
|
||||||
fn random_action(board) {
|
|
||||||
let symb = rand_symb();
|
|
||||||
let pos = rand_int(0, 10);
|
|
||||||
let action = Action(symb, pos);
|
|
||||||
|
|
||||||
while !board.can_play(action) {
|
|
||||||
let symb = rand_symb();
|
|
||||||
let pos = rand_int(0, 10);
|
|
||||||
action = Action(symb, pos);
|
|
||||||
}
|
|
||||||
|
|
||||||
return action
|
|
||||||
}
|
|
||||||
|
|
||||||
fn step_min(board) {
|
|
||||||
random_action(board)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn step_max(board) {
|
|
||||||
random_action(board)
|
|
||||||
}
|
|
||||||
"#;
|
|
||||||
|
|
||||||
let mut game = GameState::new_native(
|
|
||||||
&SCRIPT,
|
|
||||||
"max",
|
|
||||||
|_| {},
|
|
||||||
|_| {},
|
|
||||||
&SCRIPT,
|
|
||||||
"min",
|
|
||||||
|_| {},
|
|
||||||
|_| {},
|
|
||||||
|_| {},
|
|
||||||
);
|
|
||||||
|
|
||||||
let mut n = 0;
|
|
||||||
while !game.is_done() {
|
|
||||||
println!("{:?}", game.step());
|
|
||||||
println!("{:?}", game.board);
|
|
||||||
|
|
||||||
n += 1;
|
|
||||||
assert!(n < 10);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn infinite_loop() {
|
|
||||||
const SCRIPT: &str = r#"
|
|
||||||
fn step_min(board) {
|
|
||||||
loop {}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn step_max(board) {
|
|
||||||
loop {}
|
|
||||||
}
|
|
||||||
"#;
|
|
||||||
|
|
||||||
let mut game = GameState::new_native(
|
|
||||||
&SCRIPT,
|
|
||||||
"max",
|
|
||||||
|_| {},
|
|
||||||
|_| {},
|
|
||||||
&SCRIPT,
|
|
||||||
"min",
|
|
||||||
|_| {},
|
|
||||||
|_| {},
|
|
||||||
|_| {},
|
|
||||||
);
|
|
||||||
|
|
||||||
while !game.is_done() {
|
|
||||||
println!("{:?}", game.step());
|
|
||||||
println!("{:?}", game.board);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -63,14 +63,7 @@ export default function Home() {
|
|||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div>
|
<div>
|
||||||
Loading{" "}
|
Loading WASM...
|
||||||
<a
|
|
||||||
href="https://github.com/rhaiscript/playground"
|
|
||||||
target="_blank"
|
|
||||||
>
|
|
||||||
Rhai Playground
|
|
||||||
</a>
|
|
||||||
...
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,6 +1,12 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { useEffect, useRef, forwardRef, useImperativeHandle, useState } from "react";
|
import {
|
||||||
|
useEffect,
|
||||||
|
useRef,
|
||||||
|
forwardRef,
|
||||||
|
useImperativeHandle,
|
||||||
|
useState,
|
||||||
|
} from "react";
|
||||||
import styles from "@/styles/Editor.module.css";
|
import styles from "@/styles/Editor.module.css";
|
||||||
import { loadRhaiWasm, initRhaiMode } from "@/utils/wasmLoader";
|
import { loadRhaiWasm, initRhaiMode } from "@/utils/wasmLoader";
|
||||||
|
|
||||||
@@ -28,12 +34,12 @@ if (typeof window !== "undefined") {
|
|||||||
|
|
||||||
await loadRhaiWasm();
|
await loadRhaiWasm();
|
||||||
initRhaiMode(CodeMirror);
|
initRhaiMode(CodeMirror);
|
||||||
console.log('✅ WASM-based Rhai mode initialized successfully');
|
console.log("✅ WASM-based Rhai mode initialized successfully");
|
||||||
|
|
||||||
isCodeMirrorReady = true;
|
isCodeMirrorReady = true;
|
||||||
})
|
})
|
||||||
.catch(error => {
|
.catch((error) => {
|
||||||
console.error('Failed to load CodeMirror:', error);
|
console.error("Failed to load CodeMirror:", error);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -45,8 +51,8 @@ interface EditorProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const Editor = forwardRef<any, EditorProps>(function Editor(
|
export const Editor = forwardRef<any, EditorProps>(function Editor(
|
||||||
{ initialValue = "", onChange, onReady },
|
{ initialValue = "", onChange, onReady },
|
||||||
ref,
|
ref
|
||||||
) {
|
) {
|
||||||
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
||||||
const editorRef = useRef<any>(null);
|
const editorRef = useRef<any>(null);
|
||||||
@@ -56,7 +62,13 @@ export const Editor = forwardRef<any, EditorProps>(function Editor(
|
|||||||
|
|
||||||
// Initialize editor only once
|
// Initialize editor only once
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!isCodeMirrorReady || !CodeMirror || !textareaRef.current || editorRef.current) return;
|
if (
|
||||||
|
!isCodeMirrorReady ||
|
||||||
|
!CodeMirror ||
|
||||||
|
!textareaRef.current ||
|
||||||
|
editorRef.current
|
||||||
|
)
|
||||||
|
return;
|
||||||
|
|
||||||
const editor = CodeMirror.fromTextArea(textareaRef.current, {
|
const editor = CodeMirror.fromTextArea(textareaRef.current, {
|
||||||
lineNumbers: true,
|
lineNumbers: true,
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import { Button } from "@/components/ui/Button";
|
|||||||
import { Dropdown } from "@/components/ui/Dropdown";
|
import { Dropdown } from "@/components/ui/Dropdown";
|
||||||
import { Editor } from "@/components/Editor";
|
import { Editor } from "@/components/Editor";
|
||||||
import { Terminal, TerminalRef } from "@/components/Terminal";
|
import { Terminal, TerminalRef } from "@/components/Terminal";
|
||||||
import { runScript, stopScript } from "@/lib/runner";
|
import { sendDataToScript, startScript, stopScript } from "@/lib/runner";
|
||||||
import styles from "@/styles/Playground.module.css";
|
import styles from "@/styles/Playground.module.css";
|
||||||
|
|
||||||
const initialCode = `
|
const initialCode = `
|
||||||
@@ -50,17 +50,21 @@ export default function Playground() {
|
|||||||
const stopDisabled = !isScriptRunning;
|
const stopDisabled = !isScriptRunning;
|
||||||
|
|
||||||
const requestRun = useCallback(async () => {
|
const requestRun = useCallback(async () => {
|
||||||
|
if (resultRef.current) {
|
||||||
|
resultRef.current.value = "";
|
||||||
|
}
|
||||||
|
|
||||||
if (runDisabled || !editorRef.current) return;
|
if (runDisabled || !editorRef.current) return;
|
||||||
|
|
||||||
setIsScriptRunning(true);
|
setIsScriptRunning(true);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
terminalRef.current?.clear();
|
terminalRef.current?.clear();
|
||||||
|
terminalRef.current?.focus();
|
||||||
|
|
||||||
await runScript(
|
await startScript(
|
||||||
editorRef.current.getValue(),
|
editorRef.current.getValue(),
|
||||||
(line: string) => {
|
(line: string) => {
|
||||||
// Only script prints go to output text area
|
|
||||||
if (resultRef.current) {
|
if (resultRef.current) {
|
||||||
let v = resultRef.current.value + line + "\n";
|
let v = resultRef.current.value + line + "\n";
|
||||||
if (v.length > 10000) {
|
if (v.length > 10000) {
|
||||||
@@ -72,18 +76,19 @@ export default function Playground() {
|
|||||||
resultRef.current.clientHeight;
|
resultRef.current.clientHeight;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
(line: string) => {
|
(line: string) => {
|
||||||
// Game state and debug info go to terminal
|
terminalRef.current?.write(line);
|
||||||
terminalRef.current?.write(line + '\r\n');
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
} catch (ex) {
|
} catch (ex) {
|
||||||
const errorMsg = `\nEXCEPTION: "${ex}"\n`;
|
const errorMsg = `\nEXCEPTION: "${ex}"\n`;
|
||||||
if (resultRef.current) {
|
if (resultRef.current) {
|
||||||
resultRef.current.value += errorMsg;
|
resultRef.current.value += errorMsg;
|
||||||
}
|
}
|
||||||
terminalRef.current?.write('\r\n\x1B[1;31mEXCEPTION:\x1B[0m ' + String(ex) + '\r\n');
|
terminalRef.current?.write(
|
||||||
|
"\r\n\x1B[1;31mEXCEPTION:\x1B[0m " + String(ex) + "\r\n"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
setIsScriptRunning(false);
|
setIsScriptRunning(false);
|
||||||
@@ -102,9 +107,6 @@ export default function Playground() {
|
|||||||
variant="success"
|
variant="success"
|
||||||
iconLeft="play"
|
iconLeft="play"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
if (resultRef.current) {
|
|
||||||
resultRef.current.value = "";
|
|
||||||
}
|
|
||||||
requestRun();
|
requestRun();
|
||||||
}}
|
}}
|
||||||
loading={isScriptRunning}
|
loading={isScriptRunning}
|
||||||
@@ -113,6 +115,18 @@ export default function Playground() {
|
|||||||
Run
|
Run
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
variant="success"
|
||||||
|
iconLeft="play"
|
||||||
|
onClick={() => {
|
||||||
|
requestRun();
|
||||||
|
}}
|
||||||
|
loading={isScriptRunning}
|
||||||
|
disabled={runDisabled}
|
||||||
|
>
|
||||||
|
Bulk Run
|
||||||
|
</Button>
|
||||||
|
|
||||||
<Button
|
<Button
|
||||||
variant="danger"
|
variant="danger"
|
||||||
iconLeft="stop"
|
iconLeft="stop"
|
||||||
@@ -200,7 +214,10 @@ export default function Playground() {
|
|||||||
<div className={styles.terminalPanel}>
|
<div className={styles.terminalPanel}>
|
||||||
<div className={styles.panelHeader}>Terminal</div>
|
<div className={styles.panelHeader}>Terminal</div>
|
||||||
<div className={styles.terminalContainer}>
|
<div className={styles.terminalContainer}>
|
||||||
<Terminal ref={terminalRef} />
|
<Terminal
|
||||||
|
ref={terminalRef}
|
||||||
|
onData={sendDataToScript}
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className={styles.outputPanel}>
|
<div className={styles.outputPanel}>
|
||||||
|
|||||||
@@ -1,14 +1,26 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
import { useEffect, useRef, useImperativeHandle, forwardRef } from "react";
|
import {
|
||||||
import '@xterm/xterm/css/xterm.css';
|
useEffect,
|
||||||
|
useRef,
|
||||||
|
useImperativeHandle,
|
||||||
|
forwardRef,
|
||||||
|
RefObject,
|
||||||
|
} from "react";
|
||||||
|
import "@xterm/xterm/css/xterm.css";
|
||||||
|
|
||||||
export interface TerminalRef {
|
export type TerminalRef = {
|
||||||
write: (data: string) => void;
|
write: (data: string) => void;
|
||||||
clear: () => void;
|
clear: () => void;
|
||||||
}
|
focus: () => void;
|
||||||
|
};
|
||||||
|
|
||||||
export const Terminal = forwardRef<TerminalRef, {}>(function Terminal(_props, ref) {
|
export const Terminal = forwardRef<
|
||||||
|
TerminalRef,
|
||||||
|
{
|
||||||
|
onData: (data: String) => void;
|
||||||
|
}
|
||||||
|
>(function Terminal(props, ref) {
|
||||||
const terminalRef = useRef<HTMLDivElement>(null);
|
const terminalRef = useRef<HTMLDivElement>(null);
|
||||||
const xtermRef = useRef<any>(null);
|
const xtermRef = useRef<any>(null);
|
||||||
|
|
||||||
@@ -22,59 +34,34 @@ export const Terminal = forwardRef<TerminalRef, {}>(function Terminal(_props, re
|
|||||||
if (xtermRef.current) {
|
if (xtermRef.current) {
|
||||||
xtermRef.current.clear();
|
xtermRef.current.clear();
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
|
focus: () => {
|
||||||
|
if (xtermRef.current) {
|
||||||
|
xtermRef.current.focus();
|
||||||
|
}
|
||||||
|
},
|
||||||
}));
|
}));
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
// Set to false when this component is unmounted.
|
||||||
|
//
|
||||||
|
// Here's what this flag prevents:
|
||||||
|
// - <Terminal> component mounts
|
||||||
|
// - `useEffect` runs, `mounted = true`
|
||||||
|
// - `init_term()` function begins work
|
||||||
|
// - before init_term() finishes, the user navigates away and <Terminal> to unmounts
|
||||||
|
// - `useEffect` cleans up, `mounted = false`
|
||||||
|
// - `init_term()` ccompletes, and we attempt to set `xtermRef.current`, causing issues
|
||||||
let mounted = true;
|
let mounted = true;
|
||||||
|
|
||||||
const initTerminal = async () => {
|
init_term(terminalRef, props.onData, () => mounted)
|
||||||
if (!terminalRef.current) return;
|
.then((term) => {
|
||||||
|
|
||||||
try {
|
|
||||||
const { Terminal } = await import('@xterm/xterm');
|
|
||||||
|
|
||||||
if (!mounted) return;
|
if (!mounted) return;
|
||||||
|
|
||||||
const term = new Terminal({
|
|
||||||
//"fontFamily": "Fantasque",
|
|
||||||
"rows": 24,
|
|
||||||
"fontSize": 16,
|
|
||||||
"tabStopWidth": 8,
|
|
||||||
"cursorBlink": true,
|
|
||||||
"theme": {
|
|
||||||
"background": "#1D1F21",
|
|
||||||
"foreground": "#F8F8F8",
|
|
||||||
"cursor": "#F8F8F2",
|
|
||||||
"black": "#282828",
|
|
||||||
"blue": "#0087AF",
|
|
||||||
"brightBlack": "#555555",
|
|
||||||
"brightBlue": "#87DFFF",
|
|
||||||
"brightCyan": "#28D1E7",
|
|
||||||
"brightGreen": "#A8FF60",
|
|
||||||
"brightMagenta": "#985EFF",
|
|
||||||
"brightRed": "#FFAA00",
|
|
||||||
"brightWhite": "#D0D0D0",
|
|
||||||
"brightYellow": "#F1FF52",
|
|
||||||
"cyan": "#87DFEB",
|
|
||||||
"green": "#B4EC85",
|
|
||||||
"magenta": "#BD99FF",
|
|
||||||
"red": "#FF6600",
|
|
||||||
"white": "#F8F8F8",
|
|
||||||
"yellow": "#FFFFB6"
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
term.open(terminalRef.current);
|
|
||||||
term.write('Terminal ready.\r\n');
|
|
||||||
|
|
||||||
xtermRef.current = term;
|
xtermRef.current = term;
|
||||||
} catch (error) {
|
})
|
||||||
console.error('Failed to initialize terminal:', error);
|
.catch((err) => {
|
||||||
}
|
console.error("Failed to initialize terminal:", err);
|
||||||
};
|
});
|
||||||
|
|
||||||
initTerminal();
|
|
||||||
|
|
||||||
return () => {
|
return () => {
|
||||||
mounted = false;
|
mounted = false;
|
||||||
@@ -83,7 +70,57 @@ export const Terminal = forwardRef<TerminalRef, {}>(function Terminal(_props, re
|
|||||||
xtermRef.current = null;
|
xtermRef.current = null;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}, []);
|
}, [props.onData]);
|
||||||
|
|
||||||
return <div ref={terminalRef} style={{ height: "100%", width: "100%" }} />;
|
return <div ref={terminalRef} style={{ height: "100%", width: "100%" }} />;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
async function init_term(
|
||||||
|
ref: RefObject<HTMLDivElement>,
|
||||||
|
|
||||||
|
// Called when the terminal receives data
|
||||||
|
onData: (data: String) => void,
|
||||||
|
isMounted: () => boolean
|
||||||
|
) {
|
||||||
|
if (!ref.current) return;
|
||||||
|
|
||||||
|
const { Terminal } = await import("@xterm/xterm");
|
||||||
|
if (!isMounted()) return;
|
||||||
|
|
||||||
|
const term = new Terminal({
|
||||||
|
//"fontFamily": "Fantasque",
|
||||||
|
rows: 24,
|
||||||
|
fontSize: 16,
|
||||||
|
tabStopWidth: 8,
|
||||||
|
cursorBlink: false,
|
||||||
|
cursorStyle: "block",
|
||||||
|
cursorInactiveStyle: "none",
|
||||||
|
theme: {
|
||||||
|
background: "#1D1F21",
|
||||||
|
foreground: "#F8F8F8",
|
||||||
|
cursor: "#F8F8F2",
|
||||||
|
black: "#282828",
|
||||||
|
blue: "#0087AF",
|
||||||
|
brightBlack: "#555555",
|
||||||
|
brightBlue: "#87DFFF",
|
||||||
|
brightCyan: "#28D1E7",
|
||||||
|
brightGreen: "#A8FF60",
|
||||||
|
brightMagenta: "#985EFF",
|
||||||
|
brightRed: "#FFAA00",
|
||||||
|
brightWhite: "#D0D0D0",
|
||||||
|
brightYellow: "#F1FF52",
|
||||||
|
cyan: "#87DFEB",
|
||||||
|
green: "#B4EC85",
|
||||||
|
magenta: "#BD99FF",
|
||||||
|
red: "#ff2f00ff",
|
||||||
|
white: "#F8F8F8",
|
||||||
|
yellow: "#FFFFB6",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
term.open(ref.current);
|
||||||
|
term.onData(onData);
|
||||||
|
|
||||||
|
console.log("Terminal ready");
|
||||||
|
return term;
|
||||||
|
}
|
||||||
|
|||||||
@@ -2,34 +2,40 @@
|
|||||||
|
|
||||||
let worker: Worker | null = null;
|
let worker: Worker | null = null;
|
||||||
|
|
||||||
export async function runScript(
|
export function sendDataToScript(data: String) {
|
||||||
|
if (worker) {
|
||||||
|
worker.postMessage({ type: "data", data });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function startScript(
|
||||||
script: string,
|
script: string,
|
||||||
appendOutput: (line: string) => void,
|
appendOutput: (line: string) => void,
|
||||||
appendTerminal?: (line: string) => void
|
appendTerminal: (line: string) => void
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
if (worker) {
|
if (worker) {
|
||||||
worker.terminate();
|
worker.terminate();
|
||||||
}
|
}
|
||||||
|
|
||||||
worker = new Worker(new URL('./script-runner.worker.ts', import.meta.url));
|
worker = new Worker(new URL("./worker.ts", import.meta.url));
|
||||||
|
|
||||||
worker.onmessage = (event) => {
|
worker.onmessage = (event) => {
|
||||||
const { type, line, error } = event.data;
|
const { type, line, error } = event.data;
|
||||||
|
|
||||||
if (type === 'output') {
|
if (type === "output") {
|
||||||
appendOutput(line);
|
appendOutput(line);
|
||||||
} else if (type === 'terminal') {
|
} else if (type === "terminal") {
|
||||||
appendTerminal?.(line);
|
appendTerminal(line);
|
||||||
} else if (type === 'complete') {
|
} else if (type === "complete") {
|
||||||
worker?.terminate();
|
worker?.terminate();
|
||||||
worker = null;
|
worker = null;
|
||||||
resolve();
|
resolve();
|
||||||
} else if (type === 'error') {
|
} else if (type === "error") {
|
||||||
worker?.terminate();
|
worker?.terminate();
|
||||||
worker = null;
|
worker = null;
|
||||||
reject(new Error(error));
|
reject(new Error(error));
|
||||||
} else if (type === 'stopped') {
|
} else if (type === "stopped") {
|
||||||
worker?.terminate();
|
worker?.terminate();
|
||||||
worker = null;
|
worker = null;
|
||||||
resolve();
|
resolve();
|
||||||
@@ -42,12 +48,12 @@ export async function runScript(
|
|||||||
reject(error);
|
reject(error);
|
||||||
};
|
};
|
||||||
|
|
||||||
worker.postMessage({ type: 'run', script });
|
worker.postMessage({ type: "run", script });
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export function stopScript(): void {
|
export function stopScript(): void {
|
||||||
if (worker) {
|
if (worker) {
|
||||||
worker.postMessage({ type: 'stop' });
|
worker.postMessage({ type: "stop" });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import init, { GameState } from "./wasm/script_runner";
|
import init, { GameState, GameStateHuman } from "../wasm/runner";
|
||||||
|
|
||||||
let wasmReady = false;
|
let wasmReady = false;
|
||||||
let wasmInitPromise: Promise<void> | null = null;
|
let wasmInitPromise: Promise<void> | null = null;
|
||||||
let currentGame: GameState | null = null;
|
let currentGame: GameStateHuman | null = null;
|
||||||
|
|
||||||
async function initWasm(): Promise<void> {
|
async function initWasm(): Promise<void> {
|
||||||
if (wasmReady) return;
|
if (wasmReady) return;
|
||||||
@@ -20,9 +20,21 @@ async function initWasm(): Promise<void> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
self.onmessage = async (event) => {
|
self.onmessage = async (event) => {
|
||||||
const { type, script } = event.data;
|
const { type, ...event_data } = event.data;
|
||||||
|
|
||||||
if (type === "init") {
|
if (type === "data") {
|
||||||
|
if (currentGame !== null) {
|
||||||
|
currentGame.take_input(event_data.data);
|
||||||
|
|
||||||
|
if (currentGame.is_error()) {
|
||||||
|
currentGame = null;
|
||||||
|
self.postMessage({ type: "complete" });
|
||||||
|
} else if (currentGame.is_done()) {
|
||||||
|
currentGame = null;
|
||||||
|
self.postMessage({ type: "complete" });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (type === "init") {
|
||||||
try {
|
try {
|
||||||
await initWasm();
|
await initWasm();
|
||||||
self.postMessage({ type: "ready" });
|
self.postMessage({ type: "ready" });
|
||||||
@@ -37,24 +49,33 @@ self.onmessage = async (event) => {
|
|||||||
self.postMessage({ type: "output", line });
|
self.postMessage({ type: "output", line });
|
||||||
};
|
};
|
||||||
|
|
||||||
const appendDebug = (line: string) => {
|
|
||||||
self.postMessage({ type: "output", line }); // debug also goes to output
|
|
||||||
};
|
|
||||||
|
|
||||||
const appendTerminal = (line: string) => {
|
const appendTerminal = (line: string) => {
|
||||||
self.postMessage({ type: "terminal", line });
|
self.postMessage({ type: "terminal", line });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
currentGame = new GameStateHuman(
|
||||||
|
true,
|
||||||
|
event_data.script,
|
||||||
|
"Agent",
|
||||||
|
appendOutput,
|
||||||
|
appendOutput,
|
||||||
|
appendTerminal
|
||||||
|
);
|
||||||
|
|
||||||
|
currentGame.print_start();
|
||||||
|
|
||||||
|
/*
|
||||||
currentGame = new GameState(
|
currentGame = new GameState(
|
||||||
script,
|
event_data.script,
|
||||||
"max",
|
"max",
|
||||||
appendOutput,
|
appendOutput,
|
||||||
appendDebug,
|
appendOutput,
|
||||||
|
|
||||||
script,
|
// TODO: pick opponent
|
||||||
|
event_data.script,
|
||||||
"min",
|
"min",
|
||||||
appendOutput,
|
appendOutput,
|
||||||
appendDebug,
|
appendOutput,
|
||||||
|
|
||||||
appendTerminal
|
appendTerminal
|
||||||
);
|
);
|
||||||
@@ -70,6 +91,7 @@ self.onmessage = async (event) => {
|
|||||||
if (currentGame) {
|
if (currentGame) {
|
||||||
self.postMessage({ type: "complete" });
|
self.postMessage({ type: "complete" });
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
self.postMessage({ type: "error", error: String(error) });
|
self.postMessage({ type: "error", error: String(error) });
|
||||||
}
|
}
|
||||||
@@ -1,84 +1,87 @@
|
|||||||
// WASM loader for Rhai CodeMirror mode
|
// WASM loader for Rhai CodeMirror mode
|
||||||
import init, { RhaiMode, init_codemirror_pass } from '@/wasm/rhai-codemirror/rhai_codemirror.js';
|
import init, {
|
||||||
|
RhaiMode,
|
||||||
|
init_codemirror_pass,
|
||||||
|
} from "@/wasm/rhai-codemirror/rhai_codemirror.js";
|
||||||
|
|
||||||
let wasmInitialized = false;
|
let wasmInitialized = false;
|
||||||
let wasmModule: any = null;
|
let wasmModule: any = null;
|
||||||
let wasmLoadPromise: Promise<any> | null = null;
|
let wasmLoadPromise: Promise<any> | null = null;
|
||||||
|
|
||||||
export const loadRhaiWasm = async () => {
|
export const loadRhaiWasm = async () => {
|
||||||
if (wasmInitialized) {
|
if (wasmInitialized) {
|
||||||
return wasmModule;
|
return wasmModule;
|
||||||
}
|
}
|
||||||
if (wasmLoadPromise) {
|
if (wasmLoadPromise) {
|
||||||
return wasmLoadPromise;
|
return wasmLoadPromise;
|
||||||
}
|
}
|
||||||
|
|
||||||
wasmLoadPromise = (async () => {
|
wasmLoadPromise = (async () => {
|
||||||
try {
|
try {
|
||||||
// Initialize the WASM module
|
// Initialize the WASM module
|
||||||
wasmModule = await init();
|
wasmModule = await init();
|
||||||
wasmInitialized = true;
|
wasmInitialized = true;
|
||||||
|
|
||||||
return wasmModule;
|
return wasmModule;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to load Rhai WASM module:', error);
|
console.error("Failed to load Rhai WASM module:", error);
|
||||||
wasmLoadPromise = null; // Reset on error
|
wasmLoadPromise = null; // Reset on error
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
})();
|
})();
|
||||||
|
|
||||||
return wasmLoadPromise;
|
return wasmLoadPromise;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const initRhaiMode = (CodeMirror: any) => {
|
export const initRhaiMode = (CodeMirror: any) => {
|
||||||
if (!wasmInitialized || !wasmModule) {
|
if (!wasmInitialized || !wasmModule) {
|
||||||
throw new Error('WASM module not loaded. Call loadRhaiWasm() first.');
|
throw new Error("WASM module not loaded. Call loadRhaiWasm() first.");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initialize CodeMirror Pass for the WASM module
|
// Initialize CodeMirror Pass for the WASM module
|
||||||
init_codemirror_pass(CodeMirror.Pass);
|
init_codemirror_pass(CodeMirror.Pass);
|
||||||
|
|
||||||
// Define the Rhai mode using the WASM-based RhaiMode
|
// Define the Rhai mode using the WASM-based RhaiMode
|
||||||
CodeMirror.defineMode("rhai", (config: any) => {
|
CodeMirror.defineMode("rhai", (config: any) => {
|
||||||
return new RhaiMode(config.indentUnit || 4);
|
return new RhaiMode(config.indentUnit || 4);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
// Function to preload all WASM modules used by the application
|
// Function to preload all WASM modules used by the application
|
||||||
export const loadAllWasm = async (): Promise<void> => {
|
export const loadAllWasm = async (): Promise<void> => {
|
||||||
try {
|
try {
|
||||||
// Load Rhai CodeMirror WASM
|
// Load Rhai CodeMirror WASM
|
||||||
await loadRhaiWasm();
|
await loadRhaiWasm();
|
||||||
|
|
||||||
// Load Script Runner WASM by creating and immediately terminating a worker
|
// Load Script Runner WASM by creating and immediately terminating a worker
|
||||||
const worker = new Worker(new URL('../lib/script-runner.worker.ts', import.meta.url));
|
const worker = new Worker(new URL("../lib/worker.ts", import.meta.url));
|
||||||
await new Promise<void>((resolve, reject) => {
|
await new Promise<void>((resolve, reject) => {
|
||||||
const timeout = setTimeout(() => {
|
const timeout = setTimeout(() => {
|
||||||
worker.terminate();
|
worker.terminate();
|
||||||
reject(new Error('Script runner WASM load timeout'));
|
reject(new Error("Script runner WASM load timeout"));
|
||||||
}, 10000);
|
}, 10000);
|
||||||
|
|
||||||
worker.postMessage({ type: 'init' });
|
worker.postMessage({ type: "init" });
|
||||||
worker.onmessage = (event) => {
|
worker.onmessage = (event) => {
|
||||||
if (event.data.type === 'ready') {
|
if (event.data.type === "ready") {
|
||||||
clearTimeout(timeout);
|
clearTimeout(timeout);
|
||||||
worker.terminate();
|
worker.terminate();
|
||||||
resolve();
|
resolve();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
worker.onerror = (error) => {
|
worker.onerror = (error) => {
|
||||||
clearTimeout(timeout);
|
clearTimeout(timeout);
|
||||||
worker.terminate();
|
worker.terminate();
|
||||||
reject(error);
|
reject(error);
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
console.log('✅ All WASM modules loaded successfully');
|
console.log("✅ All WASM modules loaded successfully");
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('❌ Failed to load WASM modules:', error);
|
console.error("❌ Failed to load WASM modules:", error);
|
||||||
throw error;
|
throw error;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export { RhaiMode };
|
export { RhaiMode };
|
||||||
|
|||||||
Reference in New Issue
Block a user