@devaloop/devalang 0.0.1-beta.2 → 0.0.1-beta.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/Cargo.toml +84 -81
- package/README.md +3 -2
- package/docs/CHANGELOG.md +41 -0
- package/docs/ROADMAP.md +3 -3
- package/examples/chain.deva +19 -0
- package/examples/plugin.deva +10 -10
- package/examples/routing.deva +23 -0
- package/out-tsc/bin/project-version.json +6 -0
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -8
- package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
- package/out-tsc/scripts/version/copy-to-binary.js +79 -0
- package/package.json +23 -10
- package/project-version.json +3 -3
- package/rust/bindings/Cargo.toml +9 -0
- package/rust/bindings/src/lib.rs +86 -0
- package/rust/cli/addon/commands.rs +35 -0
- package/rust/cli/addon/download.rs +234 -0
- package/rust/cli/addon/install.rs +33 -0
- package/rust/cli/addon/list.rs +224 -0
- package/rust/cli/addon/metadata.rs +124 -0
- package/rust/cli/addon/mod.rs +8 -0
- package/rust/cli/addon/remove.rs +271 -0
- package/rust/cli/addon/update.rs +305 -0
- package/rust/cli/{install/addon.rs → addon/utils.rs} +109 -118
- package/rust/cli/build/commands.rs +153 -153
- package/rust/cli/build/process.rs +165 -165
- package/rust/cli/check/mod.rs +208 -208
- package/rust/cli/discover/commands.rs +275 -253
- package/rust/cli/discover/config.rs +109 -111
- package/rust/cli/discover/fs.rs +19 -19
- package/rust/cli/discover/install.rs +214 -103
- package/rust/cli/discover/metadata.rs +48 -48
- package/rust/cli/discover/mod.rs +5 -5
- package/rust/cli/me/commands.rs +52 -0
- package/rust/cli/me/mod.rs +1 -0
- package/rust/cli/mod.rs +12 -12
- package/rust/cli/parser.rs +30 -69
- package/rust/cli/play/commands.rs +375 -375
- package/rust/cli/play/process.rs +159 -159
- package/rust/core/audio/engine/driver.rs +19 -2
- package/rust/core/audio/engine/export.rs +169 -169
- package/rust/core/audio/engine/mod.rs +56 -56
- package/rust/core/audio/engine/notes/dsp.rs +88 -85
- package/rust/core/audio/engine/notes/mod.rs +53 -44
- package/rust/core/audio/engine/notes/params.rs +294 -294
- package/rust/core/audio/engine/sample/insert.rs +148 -47
- package/rust/core/audio/engine/sample/mod.rs +40 -40
- package/rust/core/audio/engine/sample/padding.rs +170 -170
- package/rust/core/audio/evaluator/condition.rs +61 -61
- package/rust/core/audio/evaluator/numeric.rs +152 -152
- package/rust/core/audio/evaluator/rhs.rs +16 -16
- package/rust/core/audio/evaluator/string_expr.rs +94 -94
- package/rust/core/audio/interpreter/driver.rs +574 -574
- package/rust/core/audio/interpreter/mod.rs +2 -2
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +9 -5
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -384
- package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +1 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +66 -11
- package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -3
- package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -192
- package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -24
- package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -116
- package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -97
- package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -100
- package/rust/core/audio/interpreter/statements/automate.rs +16 -16
- package/rust/core/audio/interpreter/statements/call.rs +31 -1
- package/rust/core/audio/interpreter/statements/condition.rs +72 -72
- package/rust/core/audio/interpreter/statements/function.rs +24 -24
- package/rust/core/audio/interpreter/statements/let_.rs +36 -36
- package/rust/core/audio/interpreter/statements/load.rs +17 -17
- package/rust/core/audio/interpreter/statements/loop_.rs +115 -115
- package/rust/core/audio/interpreter/statements/spawn.rs +51 -2
- package/rust/core/audio/interpreter/statements/trigger.rs +242 -239
- package/rust/core/audio/loader/trigger.rs +98 -98
- package/rust/core/audio/player.rs +70 -70
- package/rust/core/audio/special/mod.rs +9 -9
- package/rust/core/builder/mod.rs +129 -129
- package/rust/core/debugger/lexer.rs +27 -27
- package/rust/core/debugger/logs.rs +52 -52
- package/rust/core/debugger/preprocessor.rs +27 -27
- package/rust/core/debugger/store.rs +38 -38
- package/rust/core/lexer/driver.rs +59 -59
- package/rust/core/lexer/handler/arrow.rs +82 -82
- package/rust/core/lexer/handler/at.rs +21 -21
- package/rust/core/lexer/handler/brace.rs +41 -41
- package/rust/core/lexer/handler/colon.rs +21 -21
- package/rust/core/lexer/handler/comment.rs +30 -30
- package/rust/core/lexer/handler/dot.rs +21 -21
- package/rust/core/lexer/handler/driver.rs +337 -337
- package/rust/core/lexer/handler/identifier.rs +47 -47
- package/rust/core/lexer/handler/indent.rs +66 -66
- package/rust/core/lexer/handler/mod.rs +15 -15
- package/rust/core/lexer/handler/newline.rs +23 -23
- package/rust/core/lexer/handler/number.rs +31 -31
- package/rust/core/lexer/handler/operator.rs +46 -46
- package/rust/core/lexer/handler/parenthesis.rs +41 -41
- package/rust/core/lexer/handler/slash.rs +21 -21
- package/rust/core/lexer/handler/string.rs +63 -63
- package/rust/core/lexer/mod.rs +3 -3
- package/rust/core/mod.rs +9 -9
- package/rust/core/parser/driver/block.rs +111 -111
- package/rust/core/parser/driver/cursor.rs +82 -82
- package/rust/core/parser/driver/driver_impl.rs +21 -1
- package/rust/core/parser/driver/mod.rs +6 -6
- package/rust/core/parser/driver/parse_array.rs +120 -120
- package/rust/core/parser/driver/parse_map.rs +247 -223
- package/rust/core/parser/driver/parser.rs +160 -160
- package/rust/core/parser/handler/arrow_call.rs +65 -14
- package/rust/core/parser/handler/identifier/synth.rs +171 -135
- package/rust/core/parser/handler/mod.rs +9 -9
- package/rust/core/parser/handler/pattern.rs +24 -1
- package/rust/core/plugin/loader.rs +137 -137
- package/rust/core/plugin/mod.rs +2 -2
- package/rust/core/plugin/runner/non_wasm.rs +481 -297
- package/rust/core/plugin/runner/wasm32.rs +1 -0
- package/rust/core/preprocessor/loader/inject.rs +313 -278
- package/rust/core/preprocessor/loader/loader_helpers.rs +110 -110
- package/rust/core/preprocessor/loader/mod.rs +235 -235
- package/rust/core/preprocessor/module.rs +55 -55
- package/rust/core/preprocessor/processor/handlers.rs +107 -107
- package/rust/core/preprocessor/resolver/bank.rs +49 -49
- package/rust/core/preprocessor/resolver/call.rs +124 -124
- package/rust/core/preprocessor/resolver/condition.rs +95 -95
- package/rust/core/preprocessor/resolver/driver.rs +324 -324
- package/rust/core/preprocessor/resolver/function.rs +69 -69
- package/rust/core/preprocessor/resolver/group.rs +122 -122
- package/rust/core/preprocessor/resolver/let_.rs +32 -32
- package/rust/core/preprocessor/resolver/loop_.rs +318 -318
- package/rust/core/preprocessor/resolver/mod.rs +16 -16
- package/rust/core/preprocessor/resolver/pattern.rs +95 -83
- package/rust/core/preprocessor/resolver/spawn.rs +99 -99
- package/rust/core/preprocessor/resolver/synth.rs +54 -54
- package/rust/core/preprocessor/resolver/tempo.rs +48 -48
- package/rust/core/preprocessor/resolver/trigger.rs +116 -116
- package/rust/core/preprocessor/resolver/value.rs +176 -176
- package/rust/core/store/global.rs +57 -57
- package/rust/lib.rs +323 -323
- package/rust/macros/Cargo.toml +14 -0
- package/rust/macros/src/lib.rs +52 -0
- package/rust/main.rs +311 -142
- package/rust/types/Cargo.toml +1 -1
- package/rust/types/src/addons.rs +3 -1
- package/rust/types/src/config.rs +1 -3
- package/rust/utils/Cargo.toml +5 -2
- package/rust/utils/src/file.rs +397 -14
- package/rust/utils/src/path.rs +31 -2
- package/rust/utils/src/version.rs +38 -7
- package/rust/web/auth.rs +5 -0
- package/rust/web/forge.rs +5 -0
- package/rust/web/mod.rs +5 -3
- package/typescript/scripts/version/copy-to-binary.ts +82 -0
- package/rust/cli/bank/api.rs +0 -122
- package/rust/cli/bank/commands.rs +0 -306
- package/rust/cli/bank/mod.rs +0 -29
- package/rust/cli/install/bank.rs +0 -72
- package/rust/cli/install/commands.rs +0 -35
- package/rust/cli/install/mod.rs +0 -4
- package/rust/cli/install/plugin.rs +0 -80
package/rust/lib.rs
CHANGED
|
@@ -1,323 +1,323 @@
|
|
|
1
|
-
pub mod config;
|
|
2
|
-
pub mod core;
|
|
3
|
-
|
|
4
|
-
use crate::core::{
|
|
5
|
-
audio::{engine::AudioEngine, interpreter::driver::run_audio_program},
|
|
6
|
-
parser::statement::{Statement, StatementKind},
|
|
7
|
-
preprocessor::loader::ModuleLoader,
|
|
8
|
-
store::global::GlobalStore,
|
|
9
|
-
};
|
|
10
|
-
use devalang_types::{FunctionTable, Value, VariableTable};
|
|
11
|
-
use devalang_utils::path::normalize_path;
|
|
12
|
-
use serde::{Deserialize, Serialize};
|
|
13
|
-
use serde_wasm_bindgen::to_value;
|
|
14
|
-
use wasm_bindgen::prelude::*;
|
|
15
|
-
|
|
16
|
-
#[derive(Serialize, Deserialize)]
|
|
17
|
-
struct ParseResult {
|
|
18
|
-
ok: bool,
|
|
19
|
-
ast: String,
|
|
20
|
-
errors: Vec<ErrorResult>,
|
|
21
|
-
}
|
|
22
|
-
|
|
23
|
-
#[derive(Serialize, Deserialize)]
|
|
24
|
-
struct ErrorResult {
|
|
25
|
-
message: String,
|
|
26
|
-
line: usize,
|
|
27
|
-
column: usize,
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
#[wasm_bindgen]
|
|
31
|
-
pub fn parse(entry_path: &str, source: &str) -> Result<JsValue, JsValue> {
|
|
32
|
-
let statements = parse_internal_from_string(entry_path, source);
|
|
33
|
-
|
|
34
|
-
match statements {
|
|
35
|
-
Ok(value) => {
|
|
36
|
-
let ast_string = value;
|
|
37
|
-
to_value(&ast_string)
|
|
38
|
-
.map_err(|e| JsValue::from_str(&format!("Error converting AST to JS value: {}", e)))
|
|
39
|
-
}
|
|
40
|
-
Err(e) => Err(JsValue::from_str(&format!("Error: {}", e))),
|
|
41
|
-
}
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
#[wasm_bindgen]
|
|
45
|
-
pub fn debug_render(user_code: &str) -> Result<JsValue, JsValue> {
|
|
46
|
-
console_error_panic_hook::set_once();
|
|
47
|
-
|
|
48
|
-
let entry_path = normalize_path("playground.deva");
|
|
49
|
-
let output_path = normalize_path("./temp");
|
|
50
|
-
|
|
51
|
-
let mut global_store = GlobalStore::new();
|
|
52
|
-
|
|
53
|
-
let loader =
|
|
54
|
-
ModuleLoader::from_raw_source(&entry_path, &output_path, user_code, &mut global_store);
|
|
55
|
-
|
|
56
|
-
loader
|
|
57
|
-
.load_wasm_module(&mut global_store)
|
|
58
|
-
.map_err(|e| JsValue::from_str(&format!("Module loading error: {}", e)))?;
|
|
59
|
-
|
|
60
|
-
let all_statements_map = loader.extract_statements_map(&global_store);
|
|
61
|
-
|
|
62
|
-
let main_statements = all_statements_map
|
|
63
|
-
.get(&entry_path)
|
|
64
|
-
.ok_or(JsValue::from_str("No statements found for entry module"))?
|
|
65
|
-
.clone();
|
|
66
|
-
|
|
67
|
-
let mut audio_engine = AudioEngine::new("wasm_output".to_string());
|
|
68
|
-
|
|
69
|
-
let _ = run_audio_program(
|
|
70
|
-
&main_statements,
|
|
71
|
-
&mut audio_engine,
|
|
72
|
-
"playground".to_string(),
|
|
73
|
-
"wasm_output".to_string(),
|
|
74
|
-
VariableTable::new(),
|
|
75
|
-
FunctionTable::new(),
|
|
76
|
-
&mut global_store,
|
|
77
|
-
);
|
|
78
|
-
|
|
79
|
-
// Inspect buffer to detect if any audio was produced. In test/CI
|
|
80
|
-
// environments it's common to produce no audio (silent program);
|
|
81
|
-
// callers rely on this flag for diagnostics.
|
|
82
|
-
let samples = audio_engine.get_normalized_buffer();
|
|
83
|
-
let any_nonzero = samples.iter().any(|&s| s != 0.0);
|
|
84
|
-
|
|
85
|
-
// Build parsed AST for diagnostics
|
|
86
|
-
let ast_res = parse_internal_from_string("playground.deva", user_code);
|
|
87
|
-
let ast_str = match ast_res {
|
|
88
|
-
Ok(p) => p.ast,
|
|
89
|
-
Err(_) => "".to_string(),
|
|
90
|
-
};
|
|
91
|
-
|
|
92
|
-
#[derive(Serialize)]
|
|
93
|
-
struct DebugResult {
|
|
94
|
-
samples_len: usize,
|
|
95
|
-
any_nonzero: bool,
|
|
96
|
-
ast: String,
|
|
97
|
-
note_count: usize,
|
|
98
|
-
global_vars: Vec<String>,
|
|
99
|
-
statements_count: usize,
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
let out = DebugResult {
|
|
103
|
-
samples_len: samples.len(),
|
|
104
|
-
any_nonzero,
|
|
105
|
-
ast: ast_str,
|
|
106
|
-
note_count: audio_engine.note_count,
|
|
107
|
-
global_vars: global_store.variables.variables.keys().cloned().collect(),
|
|
108
|
-
statements_count: main_statements.len(),
|
|
109
|
-
};
|
|
110
|
-
|
|
111
|
-
to_value(&out).map_err(|e| JsValue::from_str(&format!("Error converting debug result: {}", e)))
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
#[wasm_bindgen]
|
|
115
|
-
pub fn render_audio(user_code: &str) -> Result<js_sys::Float32Array, JsValue> {
|
|
116
|
-
console_error_panic_hook::set_once();
|
|
117
|
-
|
|
118
|
-
let entry_path = normalize_path("playground.deva");
|
|
119
|
-
let output_path = normalize_path("./temp");
|
|
120
|
-
|
|
121
|
-
let mut global_store = GlobalStore::new();
|
|
122
|
-
|
|
123
|
-
let loader =
|
|
124
|
-
ModuleLoader::from_raw_source(&entry_path, &output_path, user_code, &mut global_store);
|
|
125
|
-
|
|
126
|
-
loader
|
|
127
|
-
.load_wasm_module(&mut global_store)
|
|
128
|
-
.map_err(|e| JsValue::from_str(&format!("Module loading error: {}", e)))?;
|
|
129
|
-
|
|
130
|
-
let all_statements_map = loader.extract_statements_map(&global_store);
|
|
131
|
-
|
|
132
|
-
let main_statements = all_statements_map
|
|
133
|
-
.get(&entry_path)
|
|
134
|
-
.ok_or(JsValue::from_str("No statements found for entry module"))?
|
|
135
|
-
.clone();
|
|
136
|
-
|
|
137
|
-
let mut audio_engine = AudioEngine::new("wasm_output".to_string());
|
|
138
|
-
|
|
139
|
-
let _ = run_audio_program(
|
|
140
|
-
&main_statements,
|
|
141
|
-
&mut audio_engine,
|
|
142
|
-
"playground".to_string(),
|
|
143
|
-
"wasm_output".to_string(),
|
|
144
|
-
VariableTable::new(),
|
|
145
|
-
FunctionTable::new(),
|
|
146
|
-
&mut global_store,
|
|
147
|
-
);
|
|
148
|
-
|
|
149
|
-
let samples = audio_engine.get_normalized_buffer();
|
|
150
|
-
|
|
151
|
-
if samples.is_empty() {
|
|
152
|
-
// For test environments where no audio was scheduled, return a small
|
|
153
|
-
// silent buffer instead of failing. This helps tests proceed in CI.
|
|
154
|
-
let silent = vec![0.0f32; 1024];
|
|
155
|
-
return Ok(js_sys::Float32Array::from(silent.as_slice()));
|
|
156
|
-
}
|
|
157
|
-
|
|
158
|
-
Ok(js_sys::Float32Array::from(samples.as_slice()))
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
#[wasm_bindgen]
|
|
162
|
-
#[allow(unused_variables)]
|
|
163
|
-
pub fn register_playhead_callback(cb: &js_sys::Function) {
|
|
164
|
-
// Register a JS callback to receive playhead events during real-time
|
|
165
|
-
// playback. This is a no-op on non-wasm targets to keep the bindings
|
|
166
|
-
// portable for native builds.
|
|
167
|
-
// Only register if target supports wasm callbacks
|
|
168
|
-
#[cfg(target_arch = "wasm32")]
|
|
169
|
-
{
|
|
170
|
-
crate::core::audio::interpreter::driver::register_playhead_callback(cb.clone());
|
|
171
|
-
}
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
#[wasm_bindgen]
|
|
175
|
-
#[allow(unused_variables)]
|
|
176
|
-
pub fn collect_playhead_events() -> Result<JsValue, JsValue> {
|
|
177
|
-
#[cfg(target_arch = "wasm32")]
|
|
178
|
-
{
|
|
179
|
-
let events = crate::core::audio::interpreter::driver::collect_playhead_events();
|
|
180
|
-
to_value(&events).map_err(|e| JsValue::from_str(&format!("Error converting events: {}", e)))
|
|
181
|
-
}
|
|
182
|
-
#[cfg(not(target_arch = "wasm32"))]
|
|
183
|
-
{
|
|
184
|
-
// On non-wasm targets, return an empty array
|
|
185
|
-
to_value(&Vec::<String>::new()).map_err(|e| JsValue::from_str(&format!("Error: {}", e)))
|
|
186
|
-
}
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
#[wasm_bindgen]
|
|
190
|
-
pub fn unregister_playhead_callback() {
|
|
191
|
-
#[cfg(target_arch = "wasm32")]
|
|
192
|
-
{
|
|
193
|
-
crate::core::audio::interpreter::driver::unregister_playhead_callback();
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
fn parse_internal_from_string(virtual_path: &str, source: &str) -> Result<ParseResult, String> {
|
|
198
|
-
let entry_path = normalize_path(virtual_path);
|
|
199
|
-
let output_path = normalize_path("./temp");
|
|
200
|
-
|
|
201
|
-
let mut global_store = GlobalStore::new();
|
|
202
|
-
let loader =
|
|
203
|
-
ModuleLoader::from_raw_source(&entry_path, &output_path, source, &mut global_store);
|
|
204
|
-
|
|
205
|
-
let module = loader
|
|
206
|
-
.load_single_module(&mut global_store)
|
|
207
|
-
.map_err(|e| format!("Error loading module: {}", e))?;
|
|
208
|
-
|
|
209
|
-
let raw_ast = ast_to_string(module.statements.clone());
|
|
210
|
-
|
|
211
|
-
let found_errors = collect_errors_recursively(&module.statements);
|
|
212
|
-
|
|
213
|
-
let result = ParseResult {
|
|
214
|
-
ok: true,
|
|
215
|
-
ast: raw_ast,
|
|
216
|
-
errors: found_errors,
|
|
217
|
-
};
|
|
218
|
-
|
|
219
|
-
Ok(result)
|
|
220
|
-
}
|
|
221
|
-
|
|
222
|
-
fn collect_errors_recursively(statements: &[Statement]) -> Vec<ErrorResult> {
|
|
223
|
-
let mut errors: Vec<ErrorResult> = Vec::new();
|
|
224
|
-
|
|
225
|
-
for stmt in statements {
|
|
226
|
-
match &stmt.kind {
|
|
227
|
-
StatementKind::Unknown => {
|
|
228
|
-
errors.push(ErrorResult {
|
|
229
|
-
message: format!("Unknown statement at line {}:{}", stmt.line, stmt.column),
|
|
230
|
-
line: stmt.line,
|
|
231
|
-
column: stmt.column,
|
|
232
|
-
});
|
|
233
|
-
}
|
|
234
|
-
StatementKind::Error { message } => {
|
|
235
|
-
errors.push(ErrorResult {
|
|
236
|
-
message: message.clone(),
|
|
237
|
-
line: stmt.line,
|
|
238
|
-
column: stmt.column,
|
|
239
|
-
});
|
|
240
|
-
}
|
|
241
|
-
StatementKind::Loop => {
|
|
242
|
-
if let Some(body_statements) = extract_loop_body_statements(&stmt.value) {
|
|
243
|
-
errors.extend(collect_errors_recursively(body_statements));
|
|
244
|
-
}
|
|
245
|
-
}
|
|
246
|
-
_ => {}
|
|
247
|
-
}
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
errors
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
fn extract_loop_body_statements(value: &Value) -> Option<&[Statement]> {
|
|
254
|
-
if let Value::Map(map) = value {
|
|
255
|
-
if let Some(Value::Block(statements)) = map.get("body") {
|
|
256
|
-
return Some(statements);
|
|
257
|
-
}
|
|
258
|
-
}
|
|
259
|
-
None
|
|
260
|
-
}
|
|
261
|
-
|
|
262
|
-
fn ast_to_string(statements: Vec<Statement>) -> String {
|
|
263
|
-
serde_json::to_string_pretty(&statements).expect("Failed to serialize AST")
|
|
264
|
-
}
|
|
265
|
-
|
|
266
|
-
#[cfg(test)]
|
|
267
|
-
mod tests {
|
|
268
|
-
use super::*;
|
|
269
|
-
use devalang_types::{Statement, StatementKind, Value};
|
|
270
|
-
|
|
271
|
-
#[test]
|
|
272
|
-
fn test_extract_loop_body_statements_none() {
|
|
273
|
-
let v = Value::Map(std::collections::HashMap::new());
|
|
274
|
-
assert!(extract_loop_body_statements(&v).is_none());
|
|
275
|
-
}
|
|
276
|
-
|
|
277
|
-
#[test]
|
|
278
|
-
fn test_extract_loop_body_statements_some() {
|
|
279
|
-
let stmt = Statement::unknown();
|
|
280
|
-
let mut map = std::collections::HashMap::new();
|
|
281
|
-
map.insert("body".to_string(), Value::Block(vec![stmt.clone(), stmt]));
|
|
282
|
-
|
|
283
|
-
let v = Value::Map(map);
|
|
284
|
-
let res = extract_loop_body_statements(&v);
|
|
285
|
-
assert!(res.is_some());
|
|
286
|
-
let slice = res.unwrap();
|
|
287
|
-
assert_eq!(slice.len(), 2);
|
|
288
|
-
}
|
|
289
|
-
|
|
290
|
-
#[test]
|
|
291
|
-
fn test_collect_errors_recursively_detection() {
|
|
292
|
-
let mut statements: Vec<Statement> = Vec::new();
|
|
293
|
-
|
|
294
|
-
// Unknown statement should be reported
|
|
295
|
-
let s1 = Statement::unknown_with_pos(0, 10, 2);
|
|
296
|
-
statements.push(s1.clone());
|
|
297
|
-
|
|
298
|
-
// Error statement
|
|
299
|
-
let s2 = Statement::error_with_pos(0, 20, 4, "boom".to_string());
|
|
300
|
-
statements.push(s2.clone());
|
|
301
|
-
|
|
302
|
-
// Loop with body containing unknown
|
|
303
|
-
let body_stmt = Statement::unknown_with_pos(1, 30, 5);
|
|
304
|
-
let mut loop_map = std::collections::HashMap::new();
|
|
305
|
-
loop_map.insert("body".to_string(), Value::Block(vec![body_stmt.clone()]));
|
|
306
|
-
|
|
307
|
-
let loop_stmt = Statement {
|
|
308
|
-
kind: StatementKind::Loop,
|
|
309
|
-
value: Value::Map(loop_map),
|
|
310
|
-
indent: 0,
|
|
311
|
-
line: 15,
|
|
312
|
-
column: 1,
|
|
313
|
-
};
|
|
314
|
-
statements.push(loop_stmt);
|
|
315
|
-
|
|
316
|
-
let errors = collect_errors_recursively(&statements);
|
|
317
|
-
// expect three errors: s1 unknown, s2 error, body unknown
|
|
318
|
-
assert_eq!(errors.len(), 3);
|
|
319
|
-
assert!(errors.iter().any(|e| e.line == 10));
|
|
320
|
-
assert!(errors.iter().any(|e| e.line == 20));
|
|
321
|
-
assert!(errors.iter().any(|e| e.line == 30));
|
|
322
|
-
}
|
|
323
|
-
}
|
|
1
|
+
pub mod config;
|
|
2
|
+
pub mod core;
|
|
3
|
+
|
|
4
|
+
use crate::core::{
|
|
5
|
+
audio::{engine::AudioEngine, interpreter::driver::run_audio_program},
|
|
6
|
+
parser::statement::{Statement, StatementKind},
|
|
7
|
+
preprocessor::loader::ModuleLoader,
|
|
8
|
+
store::global::GlobalStore,
|
|
9
|
+
};
|
|
10
|
+
use devalang_types::{FunctionTable, Value, VariableTable};
|
|
11
|
+
use devalang_utils::path::normalize_path;
|
|
12
|
+
use serde::{Deserialize, Serialize};
|
|
13
|
+
use serde_wasm_bindgen::to_value;
|
|
14
|
+
use wasm_bindgen::prelude::*;
|
|
15
|
+
|
|
16
|
+
#[derive(Serialize, Deserialize)]
|
|
17
|
+
struct ParseResult {
|
|
18
|
+
ok: bool,
|
|
19
|
+
ast: String,
|
|
20
|
+
errors: Vec<ErrorResult>,
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
#[derive(Serialize, Deserialize)]
|
|
24
|
+
struct ErrorResult {
|
|
25
|
+
message: String,
|
|
26
|
+
line: usize,
|
|
27
|
+
column: usize,
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
#[wasm_bindgen]
|
|
31
|
+
pub fn parse(entry_path: &str, source: &str) -> Result<JsValue, JsValue> {
|
|
32
|
+
let statements = parse_internal_from_string(entry_path, source);
|
|
33
|
+
|
|
34
|
+
match statements {
|
|
35
|
+
Ok(value) => {
|
|
36
|
+
let ast_string = value;
|
|
37
|
+
to_value(&ast_string)
|
|
38
|
+
.map_err(|e| JsValue::from_str(&format!("Error converting AST to JS value: {}", e)))
|
|
39
|
+
}
|
|
40
|
+
Err(e) => Err(JsValue::from_str(&format!("Error: {}", e))),
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
#[wasm_bindgen]
|
|
45
|
+
pub fn debug_render(user_code: &str) -> Result<JsValue, JsValue> {
|
|
46
|
+
console_error_panic_hook::set_once();
|
|
47
|
+
|
|
48
|
+
let entry_path = normalize_path("playground.deva");
|
|
49
|
+
let output_path = normalize_path("./temp");
|
|
50
|
+
|
|
51
|
+
let mut global_store = GlobalStore::new();
|
|
52
|
+
|
|
53
|
+
let loader =
|
|
54
|
+
ModuleLoader::from_raw_source(&entry_path, &output_path, user_code, &mut global_store);
|
|
55
|
+
|
|
56
|
+
loader
|
|
57
|
+
.load_wasm_module(&mut global_store)
|
|
58
|
+
.map_err(|e| JsValue::from_str(&format!("Module loading error: {}", e)))?;
|
|
59
|
+
|
|
60
|
+
let all_statements_map = loader.extract_statements_map(&global_store);
|
|
61
|
+
|
|
62
|
+
let main_statements = all_statements_map
|
|
63
|
+
.get(&entry_path)
|
|
64
|
+
.ok_or(JsValue::from_str("No statements found for entry module"))?
|
|
65
|
+
.clone();
|
|
66
|
+
|
|
67
|
+
let mut audio_engine = AudioEngine::new("wasm_output".to_string());
|
|
68
|
+
|
|
69
|
+
let _ = run_audio_program(
|
|
70
|
+
&main_statements,
|
|
71
|
+
&mut audio_engine,
|
|
72
|
+
"playground".to_string(),
|
|
73
|
+
"wasm_output".to_string(),
|
|
74
|
+
VariableTable::new(),
|
|
75
|
+
FunctionTable::new(),
|
|
76
|
+
&mut global_store,
|
|
77
|
+
);
|
|
78
|
+
|
|
79
|
+
// Inspect buffer to detect if any audio was produced. In test/CI
|
|
80
|
+
// environments it's common to produce no audio (silent program);
|
|
81
|
+
// callers rely on this flag for diagnostics.
|
|
82
|
+
let samples = audio_engine.get_normalized_buffer();
|
|
83
|
+
let any_nonzero = samples.iter().any(|&s| s != 0.0);
|
|
84
|
+
|
|
85
|
+
// Build parsed AST for diagnostics
|
|
86
|
+
let ast_res = parse_internal_from_string("playground.deva", user_code);
|
|
87
|
+
let ast_str = match ast_res {
|
|
88
|
+
Ok(p) => p.ast,
|
|
89
|
+
Err(_) => "".to_string(),
|
|
90
|
+
};
|
|
91
|
+
|
|
92
|
+
#[derive(Serialize)]
|
|
93
|
+
struct DebugResult {
|
|
94
|
+
samples_len: usize,
|
|
95
|
+
any_nonzero: bool,
|
|
96
|
+
ast: String,
|
|
97
|
+
note_count: usize,
|
|
98
|
+
global_vars: Vec<String>,
|
|
99
|
+
statements_count: usize,
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
let out = DebugResult {
|
|
103
|
+
samples_len: samples.len(),
|
|
104
|
+
any_nonzero,
|
|
105
|
+
ast: ast_str,
|
|
106
|
+
note_count: audio_engine.note_count,
|
|
107
|
+
global_vars: global_store.variables.variables.keys().cloned().collect(),
|
|
108
|
+
statements_count: main_statements.len(),
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
to_value(&out).map_err(|e| JsValue::from_str(&format!("Error converting debug result: {}", e)))
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
#[wasm_bindgen]
|
|
115
|
+
pub fn render_audio(user_code: &str) -> Result<js_sys::Float32Array, JsValue> {
|
|
116
|
+
console_error_panic_hook::set_once();
|
|
117
|
+
|
|
118
|
+
let entry_path = normalize_path("playground.deva");
|
|
119
|
+
let output_path = normalize_path("./temp");
|
|
120
|
+
|
|
121
|
+
let mut global_store = GlobalStore::new();
|
|
122
|
+
|
|
123
|
+
let loader =
|
|
124
|
+
ModuleLoader::from_raw_source(&entry_path, &output_path, user_code, &mut global_store);
|
|
125
|
+
|
|
126
|
+
loader
|
|
127
|
+
.load_wasm_module(&mut global_store)
|
|
128
|
+
.map_err(|e| JsValue::from_str(&format!("Module loading error: {}", e)))?;
|
|
129
|
+
|
|
130
|
+
let all_statements_map = loader.extract_statements_map(&global_store);
|
|
131
|
+
|
|
132
|
+
let main_statements = all_statements_map
|
|
133
|
+
.get(&entry_path)
|
|
134
|
+
.ok_or(JsValue::from_str("No statements found for entry module"))?
|
|
135
|
+
.clone();
|
|
136
|
+
|
|
137
|
+
let mut audio_engine = AudioEngine::new("wasm_output".to_string());
|
|
138
|
+
|
|
139
|
+
let _ = run_audio_program(
|
|
140
|
+
&main_statements,
|
|
141
|
+
&mut audio_engine,
|
|
142
|
+
"playground".to_string(),
|
|
143
|
+
"wasm_output".to_string(),
|
|
144
|
+
VariableTable::new(),
|
|
145
|
+
FunctionTable::new(),
|
|
146
|
+
&mut global_store,
|
|
147
|
+
);
|
|
148
|
+
|
|
149
|
+
let samples = audio_engine.get_normalized_buffer();
|
|
150
|
+
|
|
151
|
+
if samples.is_empty() {
|
|
152
|
+
// For test environments where no audio was scheduled, return a small
|
|
153
|
+
// silent buffer instead of failing. This helps tests proceed in CI.
|
|
154
|
+
let silent = vec![0.0f32; 1024];
|
|
155
|
+
return Ok(js_sys::Float32Array::from(silent.as_slice()));
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
Ok(js_sys::Float32Array::from(samples.as_slice()))
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
#[wasm_bindgen]
|
|
162
|
+
#[allow(unused_variables)]
|
|
163
|
+
pub fn register_playhead_callback(cb: &js_sys::Function) {
|
|
164
|
+
// Register a JS callback to receive playhead events during real-time
|
|
165
|
+
// playback. This is a no-op on non-wasm targets to keep the bindings
|
|
166
|
+
// portable for native builds.
|
|
167
|
+
// Only register if target supports wasm callbacks
|
|
168
|
+
#[cfg(target_arch = "wasm32")]
|
|
169
|
+
{
|
|
170
|
+
crate::core::audio::interpreter::driver::register_playhead_callback(cb.clone());
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
#[wasm_bindgen]
|
|
175
|
+
#[allow(unused_variables)]
|
|
176
|
+
pub fn collect_playhead_events() -> Result<JsValue, JsValue> {
|
|
177
|
+
#[cfg(target_arch = "wasm32")]
|
|
178
|
+
{
|
|
179
|
+
let events = crate::core::audio::interpreter::driver::collect_playhead_events();
|
|
180
|
+
to_value(&events).map_err(|e| JsValue::from_str(&format!("Error converting events: {}", e)))
|
|
181
|
+
}
|
|
182
|
+
#[cfg(not(target_arch = "wasm32"))]
|
|
183
|
+
{
|
|
184
|
+
// On non-wasm targets, return an empty array
|
|
185
|
+
to_value(&Vec::<String>::new()).map_err(|e| JsValue::from_str(&format!("Error: {}", e)))
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
#[wasm_bindgen]
|
|
190
|
+
pub fn unregister_playhead_callback() {
|
|
191
|
+
#[cfg(target_arch = "wasm32")]
|
|
192
|
+
{
|
|
193
|
+
crate::core::audio::interpreter::driver::unregister_playhead_callback();
|
|
194
|
+
}
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
fn parse_internal_from_string(virtual_path: &str, source: &str) -> Result<ParseResult, String> {
|
|
198
|
+
let entry_path = normalize_path(virtual_path);
|
|
199
|
+
let output_path = normalize_path("./temp");
|
|
200
|
+
|
|
201
|
+
let mut global_store = GlobalStore::new();
|
|
202
|
+
let loader =
|
|
203
|
+
ModuleLoader::from_raw_source(&entry_path, &output_path, source, &mut global_store);
|
|
204
|
+
|
|
205
|
+
let module = loader
|
|
206
|
+
.load_single_module(&mut global_store)
|
|
207
|
+
.map_err(|e| format!("Error loading module: {}", e))?;
|
|
208
|
+
|
|
209
|
+
let raw_ast = ast_to_string(module.statements.clone());
|
|
210
|
+
|
|
211
|
+
let found_errors = collect_errors_recursively(&module.statements);
|
|
212
|
+
|
|
213
|
+
let result = ParseResult {
|
|
214
|
+
ok: true,
|
|
215
|
+
ast: raw_ast,
|
|
216
|
+
errors: found_errors,
|
|
217
|
+
};
|
|
218
|
+
|
|
219
|
+
Ok(result)
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
fn collect_errors_recursively(statements: &[Statement]) -> Vec<ErrorResult> {
|
|
223
|
+
let mut errors: Vec<ErrorResult> = Vec::new();
|
|
224
|
+
|
|
225
|
+
for stmt in statements {
|
|
226
|
+
match &stmt.kind {
|
|
227
|
+
StatementKind::Unknown => {
|
|
228
|
+
errors.push(ErrorResult {
|
|
229
|
+
message: format!("Unknown statement at line {}:{}", stmt.line, stmt.column),
|
|
230
|
+
line: stmt.line,
|
|
231
|
+
column: stmt.column,
|
|
232
|
+
});
|
|
233
|
+
}
|
|
234
|
+
StatementKind::Error { message } => {
|
|
235
|
+
errors.push(ErrorResult {
|
|
236
|
+
message: message.clone(),
|
|
237
|
+
line: stmt.line,
|
|
238
|
+
column: stmt.column,
|
|
239
|
+
});
|
|
240
|
+
}
|
|
241
|
+
StatementKind::Loop => {
|
|
242
|
+
if let Some(body_statements) = extract_loop_body_statements(&stmt.value) {
|
|
243
|
+
errors.extend(collect_errors_recursively(body_statements));
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
_ => {}
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
|
|
250
|
+
errors
|
|
251
|
+
}
|
|
252
|
+
|
|
253
|
+
fn extract_loop_body_statements(value: &Value) -> Option<&[Statement]> {
|
|
254
|
+
if let Value::Map(map) = value {
|
|
255
|
+
if let Some(Value::Block(statements)) = map.get("body") {
|
|
256
|
+
return Some(statements);
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
None
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
fn ast_to_string(statements: Vec<Statement>) -> String {
|
|
263
|
+
serde_json::to_string_pretty(&statements).expect("Failed to serialize AST")
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
#[cfg(test)]
|
|
267
|
+
mod tests {
|
|
268
|
+
use super::*;
|
|
269
|
+
use devalang_types::{Statement, StatementKind, Value};
|
|
270
|
+
|
|
271
|
+
#[test]
|
|
272
|
+
fn test_extract_loop_body_statements_none() {
|
|
273
|
+
let v = Value::Map(std::collections::HashMap::new());
|
|
274
|
+
assert!(extract_loop_body_statements(&v).is_none());
|
|
275
|
+
}
|
|
276
|
+
|
|
277
|
+
#[test]
|
|
278
|
+
fn test_extract_loop_body_statements_some() {
|
|
279
|
+
let stmt = Statement::unknown();
|
|
280
|
+
let mut map = std::collections::HashMap::new();
|
|
281
|
+
map.insert("body".to_string(), Value::Block(vec![stmt.clone(), stmt]));
|
|
282
|
+
|
|
283
|
+
let v = Value::Map(map);
|
|
284
|
+
let res = extract_loop_body_statements(&v);
|
|
285
|
+
assert!(res.is_some());
|
|
286
|
+
let slice = res.unwrap();
|
|
287
|
+
assert_eq!(slice.len(), 2);
|
|
288
|
+
}
|
|
289
|
+
|
|
290
|
+
#[test]
|
|
291
|
+
fn test_collect_errors_recursively_detection() {
|
|
292
|
+
let mut statements: Vec<Statement> = Vec::new();
|
|
293
|
+
|
|
294
|
+
// Unknown statement should be reported
|
|
295
|
+
let s1 = Statement::unknown_with_pos(0, 10, 2);
|
|
296
|
+
statements.push(s1.clone());
|
|
297
|
+
|
|
298
|
+
// Error statement
|
|
299
|
+
let s2 = Statement::error_with_pos(0, 20, 4, "boom".to_string());
|
|
300
|
+
statements.push(s2.clone());
|
|
301
|
+
|
|
302
|
+
// Loop with body containing unknown
|
|
303
|
+
let body_stmt = Statement::unknown_with_pos(1, 30, 5);
|
|
304
|
+
let mut loop_map = std::collections::HashMap::new();
|
|
305
|
+
loop_map.insert("body".to_string(), Value::Block(vec![body_stmt.clone()]));
|
|
306
|
+
|
|
307
|
+
let loop_stmt = Statement {
|
|
308
|
+
kind: StatementKind::Loop,
|
|
309
|
+
value: Value::Map(loop_map),
|
|
310
|
+
indent: 0,
|
|
311
|
+
line: 15,
|
|
312
|
+
column: 1,
|
|
313
|
+
};
|
|
314
|
+
statements.push(loop_stmt);
|
|
315
|
+
|
|
316
|
+
let errors = collect_errors_recursively(&statements);
|
|
317
|
+
// expect three errors: s1 unknown, s2 error, body unknown
|
|
318
|
+
assert_eq!(errors.len(), 3);
|
|
319
|
+
assert!(errors.iter().any(|e| e.line == 10));
|
|
320
|
+
assert!(errors.iter().any(|e| e.line == 20));
|
|
321
|
+
assert!(errors.iter().any(|e| e.line == 30));
|
|
322
|
+
}
|
|
323
|
+
}
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
[package]
|
|
2
|
+
name = "devalang_macros"
|
|
3
|
+
description = "Macros for Devalang"
|
|
4
|
+
license = "MIT"
|
|
5
|
+
version = "0.0.1"
|
|
6
|
+
edition = "2024"
|
|
7
|
+
|
|
8
|
+
[lib]
|
|
9
|
+
proc-macro = true
|
|
10
|
+
|
|
11
|
+
[dependencies]
|
|
12
|
+
proc-macro2 = "1"
|
|
13
|
+
quote = "1"
|
|
14
|
+
syn = { version = "2", features = ["full"] }
|