@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.
Files changed (159) hide show
  1. package/Cargo.toml +84 -81
  2. package/README.md +3 -2
  3. package/docs/CHANGELOG.md +41 -0
  4. package/docs/ROADMAP.md +3 -3
  5. package/examples/chain.deva +19 -0
  6. package/examples/plugin.deva +10 -10
  7. package/examples/routing.deva +23 -0
  8. package/out-tsc/bin/project-version.json +6 -0
  9. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -8
  10. package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
  11. package/out-tsc/scripts/version/copy-to-binary.js +79 -0
  12. package/package.json +23 -10
  13. package/project-version.json +3 -3
  14. package/rust/bindings/Cargo.toml +9 -0
  15. package/rust/bindings/src/lib.rs +86 -0
  16. package/rust/cli/addon/commands.rs +35 -0
  17. package/rust/cli/addon/download.rs +234 -0
  18. package/rust/cli/addon/install.rs +33 -0
  19. package/rust/cli/addon/list.rs +224 -0
  20. package/rust/cli/addon/metadata.rs +124 -0
  21. package/rust/cli/addon/mod.rs +8 -0
  22. package/rust/cli/addon/remove.rs +271 -0
  23. package/rust/cli/addon/update.rs +305 -0
  24. package/rust/cli/{install/addon.rs → addon/utils.rs} +109 -118
  25. package/rust/cli/build/commands.rs +153 -153
  26. package/rust/cli/build/process.rs +165 -165
  27. package/rust/cli/check/mod.rs +208 -208
  28. package/rust/cli/discover/commands.rs +275 -253
  29. package/rust/cli/discover/config.rs +109 -111
  30. package/rust/cli/discover/fs.rs +19 -19
  31. package/rust/cli/discover/install.rs +214 -103
  32. package/rust/cli/discover/metadata.rs +48 -48
  33. package/rust/cli/discover/mod.rs +5 -5
  34. package/rust/cli/me/commands.rs +52 -0
  35. package/rust/cli/me/mod.rs +1 -0
  36. package/rust/cli/mod.rs +12 -12
  37. package/rust/cli/parser.rs +30 -69
  38. package/rust/cli/play/commands.rs +375 -375
  39. package/rust/cli/play/process.rs +159 -159
  40. package/rust/core/audio/engine/driver.rs +19 -2
  41. package/rust/core/audio/engine/export.rs +169 -169
  42. package/rust/core/audio/engine/mod.rs +56 -56
  43. package/rust/core/audio/engine/notes/dsp.rs +88 -85
  44. package/rust/core/audio/engine/notes/mod.rs +53 -44
  45. package/rust/core/audio/engine/notes/params.rs +294 -294
  46. package/rust/core/audio/engine/sample/insert.rs +148 -47
  47. package/rust/core/audio/engine/sample/mod.rs +40 -40
  48. package/rust/core/audio/engine/sample/padding.rs +170 -170
  49. package/rust/core/audio/evaluator/condition.rs +61 -61
  50. package/rust/core/audio/evaluator/numeric.rs +152 -152
  51. package/rust/core/audio/evaluator/rhs.rs +16 -16
  52. package/rust/core/audio/evaluator/string_expr.rs +94 -94
  53. package/rust/core/audio/interpreter/driver.rs +574 -574
  54. package/rust/core/audio/interpreter/mod.rs +2 -2
  55. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +9 -5
  56. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -384
  57. package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
  58. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +1 -0
  59. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +66 -11
  60. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -3
  61. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -192
  62. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -24
  63. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -116
  64. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -97
  65. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -100
  66. package/rust/core/audio/interpreter/statements/automate.rs +16 -16
  67. package/rust/core/audio/interpreter/statements/call.rs +31 -1
  68. package/rust/core/audio/interpreter/statements/condition.rs +72 -72
  69. package/rust/core/audio/interpreter/statements/function.rs +24 -24
  70. package/rust/core/audio/interpreter/statements/let_.rs +36 -36
  71. package/rust/core/audio/interpreter/statements/load.rs +17 -17
  72. package/rust/core/audio/interpreter/statements/loop_.rs +115 -115
  73. package/rust/core/audio/interpreter/statements/spawn.rs +51 -2
  74. package/rust/core/audio/interpreter/statements/trigger.rs +242 -239
  75. package/rust/core/audio/loader/trigger.rs +98 -98
  76. package/rust/core/audio/player.rs +70 -70
  77. package/rust/core/audio/special/mod.rs +9 -9
  78. package/rust/core/builder/mod.rs +129 -129
  79. package/rust/core/debugger/lexer.rs +27 -27
  80. package/rust/core/debugger/logs.rs +52 -52
  81. package/rust/core/debugger/preprocessor.rs +27 -27
  82. package/rust/core/debugger/store.rs +38 -38
  83. package/rust/core/lexer/driver.rs +59 -59
  84. package/rust/core/lexer/handler/arrow.rs +82 -82
  85. package/rust/core/lexer/handler/at.rs +21 -21
  86. package/rust/core/lexer/handler/brace.rs +41 -41
  87. package/rust/core/lexer/handler/colon.rs +21 -21
  88. package/rust/core/lexer/handler/comment.rs +30 -30
  89. package/rust/core/lexer/handler/dot.rs +21 -21
  90. package/rust/core/lexer/handler/driver.rs +337 -337
  91. package/rust/core/lexer/handler/identifier.rs +47 -47
  92. package/rust/core/lexer/handler/indent.rs +66 -66
  93. package/rust/core/lexer/handler/mod.rs +15 -15
  94. package/rust/core/lexer/handler/newline.rs +23 -23
  95. package/rust/core/lexer/handler/number.rs +31 -31
  96. package/rust/core/lexer/handler/operator.rs +46 -46
  97. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  98. package/rust/core/lexer/handler/slash.rs +21 -21
  99. package/rust/core/lexer/handler/string.rs +63 -63
  100. package/rust/core/lexer/mod.rs +3 -3
  101. package/rust/core/mod.rs +9 -9
  102. package/rust/core/parser/driver/block.rs +111 -111
  103. package/rust/core/parser/driver/cursor.rs +82 -82
  104. package/rust/core/parser/driver/driver_impl.rs +21 -1
  105. package/rust/core/parser/driver/mod.rs +6 -6
  106. package/rust/core/parser/driver/parse_array.rs +120 -120
  107. package/rust/core/parser/driver/parse_map.rs +247 -223
  108. package/rust/core/parser/driver/parser.rs +160 -160
  109. package/rust/core/parser/handler/arrow_call.rs +65 -14
  110. package/rust/core/parser/handler/identifier/synth.rs +171 -135
  111. package/rust/core/parser/handler/mod.rs +9 -9
  112. package/rust/core/parser/handler/pattern.rs +24 -1
  113. package/rust/core/plugin/loader.rs +137 -137
  114. package/rust/core/plugin/mod.rs +2 -2
  115. package/rust/core/plugin/runner/non_wasm.rs +481 -297
  116. package/rust/core/plugin/runner/wasm32.rs +1 -0
  117. package/rust/core/preprocessor/loader/inject.rs +313 -278
  118. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -110
  119. package/rust/core/preprocessor/loader/mod.rs +235 -235
  120. package/rust/core/preprocessor/module.rs +55 -55
  121. package/rust/core/preprocessor/processor/handlers.rs +107 -107
  122. package/rust/core/preprocessor/resolver/bank.rs +49 -49
  123. package/rust/core/preprocessor/resolver/call.rs +124 -124
  124. package/rust/core/preprocessor/resolver/condition.rs +95 -95
  125. package/rust/core/preprocessor/resolver/driver.rs +324 -324
  126. package/rust/core/preprocessor/resolver/function.rs +69 -69
  127. package/rust/core/preprocessor/resolver/group.rs +122 -122
  128. package/rust/core/preprocessor/resolver/let_.rs +32 -32
  129. package/rust/core/preprocessor/resolver/loop_.rs +318 -318
  130. package/rust/core/preprocessor/resolver/mod.rs +16 -16
  131. package/rust/core/preprocessor/resolver/pattern.rs +95 -83
  132. package/rust/core/preprocessor/resolver/spawn.rs +99 -99
  133. package/rust/core/preprocessor/resolver/synth.rs +54 -54
  134. package/rust/core/preprocessor/resolver/tempo.rs +48 -48
  135. package/rust/core/preprocessor/resolver/trigger.rs +116 -116
  136. package/rust/core/preprocessor/resolver/value.rs +176 -176
  137. package/rust/core/store/global.rs +57 -57
  138. package/rust/lib.rs +323 -323
  139. package/rust/macros/Cargo.toml +14 -0
  140. package/rust/macros/src/lib.rs +52 -0
  141. package/rust/main.rs +311 -142
  142. package/rust/types/Cargo.toml +1 -1
  143. package/rust/types/src/addons.rs +3 -1
  144. package/rust/types/src/config.rs +1 -3
  145. package/rust/utils/Cargo.toml +5 -2
  146. package/rust/utils/src/file.rs +397 -14
  147. package/rust/utils/src/path.rs +31 -2
  148. package/rust/utils/src/version.rs +38 -7
  149. package/rust/web/auth.rs +5 -0
  150. package/rust/web/forge.rs +5 -0
  151. package/rust/web/mod.rs +5 -3
  152. package/typescript/scripts/version/copy-to-binary.ts +82 -0
  153. package/rust/cli/bank/api.rs +0 -122
  154. package/rust/cli/bank/commands.rs +0 -306
  155. package/rust/cli/bank/mod.rs +0 -29
  156. package/rust/cli/install/bank.rs +0 -72
  157. package/rust/cli/install/commands.rs +0 -35
  158. package/rust/cli/install/mod.rs +0 -4
  159. 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"] }