@devaloop/devalang 0.0.1-beta.1 → 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 (207) hide show
  1. package/.devalang +9 -10
  2. package/Cargo.toml +84 -80
  3. package/README.md +10 -7
  4. package/docs/CHANGELOG.md +83 -0
  5. package/docs/ROADMAP.md +6 -2
  6. package/docs/TODO.md +3 -14
  7. package/examples/bus.deva +10 -0
  8. package/examples/chain.deva +19 -0
  9. package/examples/effect.deva +2 -0
  10. package/examples/filter.deva +11 -0
  11. package/examples/lfo.deva +9 -0
  12. package/examples/plugin.deva +10 -10
  13. package/examples/routing.deva +23 -0
  14. package/examples/synth.deva +11 -1
  15. package/examples/synth_types.deva +17 -0
  16. package/out-tsc/bin/project-version.json +6 -0
  17. package/out-tsc/core/functions/index.d.ts +5 -0
  18. package/out-tsc/core/functions/index.js +11 -0
  19. package/out-tsc/pkg/devalang_core.d.ts +2 -0
  20. package/out-tsc/pkg/devalang_core.js +17 -2
  21. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +1 -0
  22. package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
  23. package/out-tsc/scripts/version/copy-to-binary.js +79 -0
  24. package/package.json +23 -10
  25. package/project-version.json +3 -3
  26. package/rust/bindings/Cargo.toml +9 -0
  27. package/rust/bindings/src/lib.rs +86 -0
  28. package/rust/cli/addon/commands.rs +35 -0
  29. package/rust/cli/addon/download.rs +234 -0
  30. package/rust/cli/addon/install.rs +33 -0
  31. package/rust/cli/addon/list.rs +224 -0
  32. package/rust/cli/addon/metadata.rs +124 -0
  33. package/rust/cli/addon/mod.rs +8 -0
  34. package/rust/cli/addon/remove.rs +271 -0
  35. package/rust/cli/addon/update.rs +305 -0
  36. package/rust/cli/{install/addon.rs → addon/utils.rs} +34 -43
  37. package/rust/cli/build/commands.rs +153 -103
  38. package/rust/cli/build/mod.rs +2 -2
  39. package/rust/cli/build/process.rs +165 -146
  40. package/rust/cli/check/mod.rs +208 -208
  41. package/rust/cli/discover/commands.rs +53 -31
  42. package/rust/cli/discover/config.rs +2 -4
  43. package/rust/cli/discover/install.rs +139 -28
  44. package/rust/cli/discover/metadata.rs +3 -3
  45. package/rust/cli/login/commands.rs +124 -124
  46. package/rust/cli/me/commands.rs +52 -0
  47. package/rust/cli/me/mod.rs +1 -0
  48. package/rust/cli/mod.rs +2 -2
  49. package/rust/cli/parser.rs +76 -70
  50. package/rust/cli/play/commands.rs +375 -324
  51. package/rust/cli/play/mod.rs +5 -5
  52. package/rust/cli/play/process.rs +159 -150
  53. package/rust/cli/play/realtime.rs +91 -91
  54. package/rust/cli/telemetry/commands.rs +22 -22
  55. package/rust/cli/telemetry/event_creator.rs +80 -80
  56. package/rust/cli/telemetry/mod.rs +3 -3
  57. package/rust/cli/telemetry/send.rs +51 -51
  58. package/rust/cli/template/commands.rs +69 -69
  59. package/rust/config/driver.rs +112 -103
  60. package/rust/config/mod.rs +3 -3
  61. package/rust/config/ops.rs +26 -26
  62. package/rust/config/settings.rs +101 -101
  63. package/rust/core/audio/engine/driver.rs +237 -0
  64. package/rust/core/audio/engine/export.rs +169 -0
  65. package/rust/core/audio/engine/helpers.rs +178 -170
  66. package/rust/core/audio/engine/mod.rs +56 -7
  67. package/rust/core/audio/engine/notes/dsp.rs +88 -0
  68. package/rust/core/audio/engine/notes/mod.rs +53 -0
  69. package/rust/core/audio/engine/notes/params.rs +294 -0
  70. package/rust/core/audio/engine/sample/insert.rs +300 -0
  71. package/rust/core/audio/engine/sample/mod.rs +40 -0
  72. package/rust/core/audio/engine/sample/padding.rs +170 -0
  73. package/rust/core/audio/evaluator/condition.rs +61 -0
  74. package/rust/core/audio/evaluator/mod.rs +9 -0
  75. package/rust/core/audio/{evaluator.rs → evaluator/numeric.rs} +152 -310
  76. package/rust/core/audio/evaluator/rhs.rs +16 -0
  77. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  78. package/rust/core/audio/interpreter/driver.rs +574 -542
  79. package/rust/core/audio/interpreter/mod.rs +2 -14
  80. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +179 -0
  81. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -0
  82. package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
  83. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +3 -0
  84. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +371 -0
  85. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  86. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  87. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  88. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  89. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  90. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  91. package/rust/core/audio/interpreter/{automate.rs → statements/automate.rs} +2 -4
  92. package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +36 -5
  93. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -71
  94. package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +24 -26
  95. package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +36 -38
  96. package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +17 -19
  97. package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +115 -114
  98. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  99. package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
  100. package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +54 -4
  101. package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
  102. package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +242 -239
  103. package/rust/core/audio/loader/trigger.rs +98 -97
  104. package/rust/core/audio/mod.rs +6 -7
  105. package/rust/core/audio/special/easing.rs +189 -189
  106. package/rust/core/audio/special/env.rs +45 -45
  107. package/rust/core/audio/special/math.rs +134 -134
  108. package/rust/core/audio/special/modulator.rs +143 -143
  109. package/rust/core/builder/mod.rs +129 -86
  110. package/rust/core/debugger/{module.rs → logs.rs} +52 -55
  111. package/rust/core/debugger/mod.rs +30 -30
  112. package/rust/core/debugger/store.rs +38 -40
  113. package/rust/core/error/mod.rs +269 -269
  114. package/rust/core/lexer/driver.rs +2 -4
  115. package/rust/core/mod.rs +9 -10
  116. package/rust/core/parser/driver/block.rs +111 -0
  117. package/rust/core/parser/driver/cursor.rs +82 -0
  118. package/rust/core/parser/driver/driver_impl.rs +159 -0
  119. package/rust/core/parser/driver/mod.rs +6 -0
  120. package/rust/core/parser/driver/parse_array.rs +120 -0
  121. package/rust/core/parser/driver/parse_map.rs +247 -0
  122. package/rust/core/parser/driver/parser.rs +160 -0
  123. package/rust/core/parser/handler/arrow_call.rs +90 -15
  124. package/rust/core/parser/handler/at.rs +279 -279
  125. package/rust/core/parser/handler/bank.rs +104 -104
  126. package/rust/core/parser/handler/condition.rs +83 -83
  127. package/rust/core/parser/handler/dot.rs +148 -148
  128. package/rust/core/parser/handler/identifier/automate.rs +254 -254
  129. package/rust/core/parser/handler/identifier/call.rs +91 -91
  130. package/rust/core/parser/handler/identifier/emit.rs +70 -70
  131. package/rust/core/parser/handler/identifier/function.rs +113 -113
  132. package/rust/core/parser/handler/identifier/group.rs +89 -89
  133. package/rust/core/parser/handler/identifier/let_.rs +173 -173
  134. package/rust/core/parser/handler/identifier/mod.rs +55 -55
  135. package/rust/core/parser/handler/identifier/on.rs +107 -107
  136. package/rust/core/parser/handler/identifier/print.rs +49 -49
  137. package/rust/core/parser/handler/identifier/sleep.rs +96 -43
  138. package/rust/core/parser/handler/identifier/spawn.rs +91 -91
  139. package/rust/core/parser/handler/identifier/synth.rs +39 -3
  140. package/rust/core/parser/handler/loop_.rs +194 -194
  141. package/rust/core/parser/handler/pattern.rs +25 -2
  142. package/rust/core/parser/handler/tempo.rs +105 -57
  143. package/rust/core/parser/statement.rs +10 -11
  144. package/rust/core/plugin/loader.rs +137 -137
  145. package/rust/core/plugin/runner/mod.rs +11 -0
  146. package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +206 -72
  147. package/rust/core/plugin/runner/wasm32.rs +44 -0
  148. package/rust/core/preprocessor/loader/inject.rs +313 -0
  149. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  150. package/rust/core/preprocessor/loader/mod.rs +235 -0
  151. package/rust/core/preprocessor/module.rs +55 -60
  152. package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +107 -114
  153. package/rust/core/preprocessor/processor/mod.rs +1 -0
  154. package/rust/core/preprocessor/resolver/function.rs +69 -69
  155. package/rust/core/preprocessor/resolver/group.rs +122 -94
  156. package/rust/core/preprocessor/resolver/pattern.rs +14 -2
  157. package/rust/core/store/global.rs +57 -61
  158. package/rust/core/store/mod.rs +1 -5
  159. package/rust/lib.rs +323 -308
  160. package/rust/macros/Cargo.toml +14 -0
  161. package/rust/macros/src/lib.rs +52 -0
  162. package/rust/main.rs +336 -143
  163. package/rust/types/Cargo.toml +1 -1
  164. package/rust/types/src/addons.rs +57 -55
  165. package/rust/types/src/config.rs +82 -74
  166. package/rust/types/src/lib.rs +15 -12
  167. package/rust/types/src/plugin.rs +20 -0
  168. package/rust/types/src/store.rs +139 -0
  169. package/rust/types/src/telemetry.rs +85 -85
  170. package/rust/utils/Cargo.toml +5 -2
  171. package/rust/utils/src/file.rs +477 -94
  172. package/rust/utils/src/first_usage.rs +97 -97
  173. package/rust/utils/src/lib.rs +9 -9
  174. package/rust/utils/src/logger.rs +200 -200
  175. package/rust/utils/src/path.rs +158 -88
  176. package/rust/utils/src/signature.rs +41 -41
  177. package/rust/utils/src/spinner.rs +20 -20
  178. package/rust/utils/src/version.rs +58 -27
  179. package/rust/utils/src/watcher.rs +46 -46
  180. package/rust/web/api.rs +5 -5
  181. package/rust/web/auth.rs +5 -0
  182. package/rust/web/cdn.rs +34 -34
  183. package/rust/web/forge.rs +5 -0
  184. package/rust/web/mod.rs +2 -0
  185. package/tests/integration.rs +21 -21
  186. package/typescript/core/functions/index.ts +11 -0
  187. package/typescript/pkg/devalang_core.ts +20 -4
  188. package/typescript/scripts/version/copy-to-binary.ts +82 -0
  189. package/rust/cli/bank/api.rs +0 -122
  190. package/rust/cli/bank/commands.rs +0 -275
  191. package/rust/cli/bank/mod.rs +0 -29
  192. package/rust/cli/install/bank.rs +0 -53
  193. package/rust/cli/install/commands.rs +0 -35
  194. package/rust/cli/install/mod.rs +0 -4
  195. package/rust/cli/install/plugin.rs +0 -61
  196. package/rust/core/audio/engine/sample.rs +0 -366
  197. package/rust/core/audio/engine/synth.rs +0 -325
  198. package/rust/core/audio/interpreter/arrow_call.rs +0 -311
  199. package/rust/core/audio/renderer.rs +0 -54
  200. package/rust/core/parser/driver.rs +0 -584
  201. package/rust/core/preprocessor/loader.rs +0 -637
  202. package/rust/core/store/export.rs +0 -28
  203. package/rust/core/store/function.rs +0 -40
  204. package/rust/core/store/import.rs +0 -28
  205. package/rust/core/store/variable.rs +0 -51
  206. package/rust/core/utils/mod.rs +0 -1
  207. package/rust/core/utils/path.rs +0 -37
package/rust/lib.rs CHANGED
@@ -1,308 +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::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
9
- utils::path::normalize_path,
10
- };
11
- use devalang_types::Value;
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
- pub fn unregister_playhead_callback() {
176
- #[cfg(target_arch = "wasm32")]
177
- {
178
- crate::core::audio::interpreter::driver::unregister_playhead_callback();
179
- }
180
- }
181
-
182
- fn parse_internal_from_string(virtual_path: &str, source: &str) -> Result<ParseResult, String> {
183
- let entry_path = normalize_path(virtual_path);
184
- let output_path = normalize_path("./temp");
185
-
186
- let mut global_store = GlobalStore::new();
187
- let loader =
188
- ModuleLoader::from_raw_source(&entry_path, &output_path, source, &mut global_store);
189
-
190
- let module = loader
191
- .load_single_module(&mut global_store)
192
- .map_err(|e| format!("Error loading module: {}", e))?;
193
-
194
- let raw_ast = ast_to_string(module.statements.clone());
195
-
196
- let found_errors = collect_errors_recursively(&module.statements);
197
-
198
- let result = ParseResult {
199
- ok: true,
200
- ast: raw_ast,
201
- errors: found_errors,
202
- };
203
-
204
- Ok(result)
205
- }
206
-
207
- fn collect_errors_recursively(statements: &[Statement]) -> Vec<ErrorResult> {
208
- let mut errors: Vec<ErrorResult> = Vec::new();
209
-
210
- for stmt in statements {
211
- match &stmt.kind {
212
- StatementKind::Unknown => {
213
- errors.push(ErrorResult {
214
- message: format!("Unknown statement at line {}:{}", stmt.line, stmt.column),
215
- line: stmt.line,
216
- column: stmt.column,
217
- });
218
- }
219
- StatementKind::Error { message } => {
220
- errors.push(ErrorResult {
221
- message: message.clone(),
222
- line: stmt.line,
223
- column: stmt.column,
224
- });
225
- }
226
- StatementKind::Loop => {
227
- if let Some(body_statements) = extract_loop_body_statements(&stmt.value) {
228
- errors.extend(collect_errors_recursively(body_statements));
229
- }
230
- }
231
- _ => {}
232
- }
233
- }
234
-
235
- errors
236
- }
237
-
238
- fn extract_loop_body_statements(value: &Value) -> Option<&[Statement]> {
239
- if let Value::Map(map) = value {
240
- if let Some(Value::Block(statements)) = map.get("body") {
241
- return Some(statements);
242
- }
243
- }
244
- None
245
- }
246
-
247
- fn ast_to_string(statements: Vec<Statement>) -> String {
248
- serde_json::to_string_pretty(&statements).expect("Failed to serialize AST")
249
- }
250
-
251
- #[cfg(test)]
252
- mod tests {
253
- use super::*;
254
- use devalang_types::{Statement, StatementKind, Value};
255
-
256
- #[test]
257
- fn test_extract_loop_body_statements_none() {
258
- let v = Value::Map(std::collections::HashMap::new());
259
- assert!(extract_loop_body_statements(&v).is_none());
260
- }
261
-
262
- #[test]
263
- fn test_extract_loop_body_statements_some() {
264
- let stmt = Statement::unknown();
265
- let mut map = std::collections::HashMap::new();
266
- map.insert("body".to_string(), Value::Block(vec![stmt.clone(), stmt]));
267
-
268
- let v = Value::Map(map);
269
- let res = extract_loop_body_statements(&v);
270
- assert!(res.is_some());
271
- let slice = res.unwrap();
272
- assert_eq!(slice.len(), 2);
273
- }
274
-
275
- #[test]
276
- fn test_collect_errors_recursively_detection() {
277
- let mut statements: Vec<Statement> = Vec::new();
278
-
279
- // Unknown statement should be reported
280
- let s1 = Statement::unknown_with_pos(0, 10, 2);
281
- statements.push(s1.clone());
282
-
283
- // Error statement
284
- let s2 = Statement::error_with_pos(0, 20, 4, "boom".to_string());
285
- statements.push(s2.clone());
286
-
287
- // Loop with body containing unknown
288
- let body_stmt = Statement::unknown_with_pos(1, 30, 5);
289
- let mut loop_map = std::collections::HashMap::new();
290
- loop_map.insert("body".to_string(), Value::Block(vec![body_stmt.clone()]));
291
-
292
- let loop_stmt = Statement {
293
- kind: StatementKind::Loop,
294
- value: Value::Map(loop_map),
295
- indent: 0,
296
- line: 15,
297
- column: 1,
298
- };
299
- statements.push(loop_stmt);
300
-
301
- let errors = collect_errors_recursively(&statements);
302
- // expect three errors: s1 unknown, s2 error, body unknown
303
- assert_eq!(errors.len(), 3);
304
- assert!(errors.iter().any(|e| e.line == 10));
305
- assert!(errors.iter().any(|e| e.line == 20));
306
- assert!(errors.iter().any(|e| e.line == 30));
307
- }
308
- }
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"] }