@devaloop/devalang 0.0.1-alpha.8 → 0.0.1-beta.1

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 (277) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +10 -4
  3. package/.github/workflows/ci.yml +103 -0
  4. package/Cargo.toml +80 -48
  5. package/README.md +135 -158
  6. package/docs/CHANGELOG.md +413 -1
  7. package/docs/CONTRIBUTING.md +101 -0
  8. package/docs/ROADMAP.md +10 -7
  9. package/docs/TODO.md +21 -9
  10. package/examples/automation.deva +42 -0
  11. package/examples/bank.deva +7 -0
  12. package/examples/condition.deva +8 -12
  13. package/examples/duration.deva +9 -0
  14. package/examples/events.deva +12 -0
  15. package/examples/function.deva +15 -0
  16. package/examples/group.deva +3 -3
  17. package/examples/index.deva +57 -10
  18. package/examples/loop.deva +7 -12
  19. package/examples/pattern.deva +8 -0
  20. package/examples/plugin.deva +16 -0
  21. package/examples/synth.deva +14 -0
  22. package/examples/variables.deva +2 -2
  23. package/out-tsc/bin/index.d.ts +2 -0
  24. package/out-tsc/bin/index.js +51 -7
  25. package/out-tsc/core/functions/index.d.ts +37 -0
  26. package/out-tsc/core/functions/index.js +76 -0
  27. package/out-tsc/core/index.d.ts +6 -0
  28. package/out-tsc/core/index.js +22 -0
  29. package/out-tsc/core/types/index.d.ts +4 -0
  30. package/out-tsc/core/types/index.js +20 -0
  31. package/out-tsc/core/types/plugin.d.ts +18 -0
  32. package/out-tsc/core/types/plugin.js +2 -0
  33. package/out-tsc/core/types/result.d.ts +27 -0
  34. package/out-tsc/core/types/result.js +2 -0
  35. package/out-tsc/core/types/statement.d.ts +106 -0
  36. package/out-tsc/core/types/statement.js +2 -0
  37. package/out-tsc/core/types/value.d.ts +43 -0
  38. package/out-tsc/core/types/value.js +2 -0
  39. package/out-tsc/index.d.ts +7 -0
  40. package/out-tsc/index.js +42 -1
  41. package/out-tsc/pkg/devalang_core.d.ts +13 -0
  42. package/out-tsc/pkg/devalang_core.js +50 -0
  43. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +33 -0
  44. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  45. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  46. package/out-tsc/scripts/postinstall.d.ts +1 -0
  47. package/out-tsc/scripts/postinstall.js +83 -0
  48. package/out-tsc/scripts/version/bump.d.ts +1 -0
  49. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  50. package/out-tsc/scripts/version/fetch.js +1 -5
  51. package/out-tsc/scripts/version/index.d.ts +1 -0
  52. package/out-tsc/scripts/version/sync.d.ts +1 -0
  53. package/package.json +28 -7
  54. package/project-version.json +4 -4
  55. package/rust/cli/bank/api.rs +122 -0
  56. package/rust/cli/bank/commands.rs +275 -0
  57. package/rust/cli/bank/mod.rs +29 -0
  58. package/rust/cli/build/commands.rs +103 -0
  59. package/rust/cli/build/mod.rs +2 -0
  60. package/rust/cli/build/process.rs +146 -0
  61. package/rust/cli/check/mod.rs +208 -0
  62. package/rust/cli/discover/commands.rs +253 -0
  63. package/rust/cli/discover/config.rs +111 -0
  64. package/rust/cli/discover/fs.rs +19 -0
  65. package/rust/cli/discover/install.rs +103 -0
  66. package/rust/cli/discover/metadata.rs +48 -0
  67. package/rust/cli/discover/mod.rs +5 -0
  68. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  69. package/rust/cli/init/mod.rs +1 -0
  70. package/rust/cli/install/addon.rs +118 -0
  71. package/rust/cli/install/bank.rs +53 -0
  72. package/rust/cli/install/commands.rs +35 -0
  73. package/rust/cli/install/mod.rs +4 -0
  74. package/rust/cli/install/plugin.rs +61 -0
  75. package/rust/cli/login/commands.rs +124 -0
  76. package/rust/cli/login/mod.rs +1 -0
  77. package/rust/cli/mod.rs +12 -205
  78. package/rust/cli/parser.rs +314 -0
  79. package/rust/cli/play/commands.rs +324 -0
  80. package/rust/cli/play/io.rs +17 -0
  81. package/rust/cli/play/mod.rs +5 -0
  82. package/rust/cli/play/process.rs +150 -0
  83. package/rust/cli/play/realtime.rs +91 -0
  84. package/rust/cli/play/utils.rs +23 -0
  85. package/rust/cli/telemetry/commands.rs +22 -0
  86. package/rust/cli/telemetry/event_creator.rs +80 -0
  87. package/rust/cli/telemetry/mod.rs +3 -0
  88. package/rust/cli/telemetry/send.rs +51 -0
  89. package/rust/cli/{template.rs → template/commands.rs} +69 -57
  90. package/rust/cli/template/mod.rs +1 -0
  91. package/rust/cli/update/commands.rs +6 -0
  92. package/rust/cli/update/mod.rs +1 -0
  93. package/rust/config/driver.rs +103 -0
  94. package/rust/config/mod.rs +3 -16
  95. package/rust/config/ops.rs +26 -0
  96. package/rust/config/settings.rs +101 -0
  97. package/rust/core/audio/engine/helpers.rs +170 -0
  98. package/rust/core/audio/engine/mod.rs +7 -0
  99. package/rust/core/audio/engine/sample.rs +366 -0
  100. package/rust/core/audio/engine/synth.rs +325 -0
  101. package/rust/core/audio/evaluator.rs +310 -31
  102. package/rust/core/audio/interpreter/arrow_call.rs +311 -0
  103. package/rust/core/audio/interpreter/automate.rs +18 -0
  104. package/rust/core/audio/interpreter/call.rs +294 -42
  105. package/rust/core/audio/interpreter/condition.rs +71 -65
  106. package/rust/core/audio/interpreter/driver.rs +542 -204
  107. package/rust/core/audio/interpreter/function.rs +26 -0
  108. package/rust/core/audio/interpreter/let_.rs +38 -19
  109. package/rust/core/audio/interpreter/load.rs +19 -18
  110. package/rust/core/audio/interpreter/loop_.rs +114 -59
  111. package/rust/core/audio/interpreter/mod.rs +14 -11
  112. package/rust/core/audio/interpreter/sleep.rs +28 -36
  113. package/rust/core/audio/interpreter/spawn.rs +252 -65
  114. package/rust/core/audio/interpreter/tempo.rs +40 -16
  115. package/rust/core/audio/interpreter/trigger.rs +239 -69
  116. package/rust/core/audio/loader/mod.rs +1 -1
  117. package/rust/core/audio/loader/trigger.rs +97 -52
  118. package/rust/core/audio/mod.rs +7 -6
  119. package/rust/core/audio/player.rs +70 -54
  120. package/rust/core/audio/renderer.rs +54 -57
  121. package/rust/core/audio/special/easing.rs +189 -0
  122. package/rust/core/audio/special/env.rs +45 -0
  123. package/rust/core/audio/special/math.rs +134 -0
  124. package/rust/core/audio/special/mod.rs +9 -0
  125. package/rust/core/audio/special/modulator.rs +143 -0
  126. package/rust/core/builder/mod.rs +86 -80
  127. package/rust/core/debugger/lexer.rs +27 -27
  128. package/rust/core/debugger/mod.rs +30 -21
  129. package/rust/core/debugger/module.rs +55 -0
  130. package/rust/core/debugger/preprocessor.rs +27 -27
  131. package/rust/core/debugger/store.rs +40 -25
  132. package/rust/core/error/mod.rs +269 -60
  133. package/rust/core/lexer/driver.rs +61 -0
  134. package/rust/core/lexer/handler/arrow.rs +82 -0
  135. package/rust/core/lexer/handler/at.rs +21 -21
  136. package/rust/core/lexer/handler/brace.rs +41 -41
  137. package/rust/core/lexer/handler/colon.rs +21 -21
  138. package/rust/core/lexer/handler/comment.rs +30 -30
  139. package/rust/core/lexer/handler/dot.rs +21 -21
  140. package/rust/core/lexer/handler/driver.rs +337 -215
  141. package/rust/core/lexer/handler/identifier.rs +47 -40
  142. package/rust/core/lexer/handler/indent.rs +66 -52
  143. package/rust/core/lexer/handler/mod.rs +15 -13
  144. package/rust/core/lexer/handler/newline.rs +23 -23
  145. package/rust/core/lexer/handler/number.rs +31 -31
  146. package/rust/core/lexer/handler/operator.rs +46 -44
  147. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  148. package/rust/core/lexer/handler/slash.rs +21 -0
  149. package/rust/core/lexer/handler/string.rs +63 -63
  150. package/rust/core/lexer/mod.rs +3 -30
  151. package/rust/core/lexer/token.rs +21 -12
  152. package/rust/core/mod.rs +10 -10
  153. package/rust/core/parser/driver.rs +584 -312
  154. package/rust/core/parser/handler/arrow_call.rs +253 -0
  155. package/rust/core/parser/handler/at.rs +279 -162
  156. package/rust/core/parser/handler/bank.rs +104 -41
  157. package/rust/core/parser/handler/condition.rs +83 -74
  158. package/rust/core/parser/handler/dot.rs +148 -112
  159. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  160. package/rust/core/parser/handler/identifier/call.rs +91 -0
  161. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  162. package/rust/core/parser/handler/identifier/function.rs +113 -0
  163. package/rust/core/parser/handler/identifier/group.rs +89 -0
  164. package/rust/core/parser/handler/identifier/let_.rs +173 -0
  165. package/rust/core/parser/handler/identifier/mod.rs +55 -0
  166. package/rust/core/parser/handler/identifier/on.rs +107 -0
  167. package/rust/core/parser/handler/identifier/print.rs +49 -0
  168. package/rust/core/parser/handler/identifier/sleep.rs +43 -0
  169. package/rust/core/parser/handler/identifier/spawn.rs +91 -0
  170. package/rust/core/parser/handler/identifier/synth.rs +135 -0
  171. package/rust/core/parser/handler/loop_.rs +194 -66
  172. package/rust/core/parser/handler/mod.rs +9 -7
  173. package/rust/core/parser/handler/pattern.rs +74 -0
  174. package/rust/core/parser/handler/tempo.rs +57 -47
  175. package/rust/core/parser/mod.rs +3 -4
  176. package/rust/core/parser/statement.rs +11 -88
  177. package/rust/core/plugin/loader.rs +137 -0
  178. package/rust/core/plugin/mod.rs +2 -0
  179. package/rust/core/plugin/runner.rs +347 -0
  180. package/rust/core/preprocessor/loader.rs +637 -179
  181. package/rust/core/preprocessor/mod.rs +4 -4
  182. package/rust/core/preprocessor/module.rs +60 -53
  183. package/rust/core/preprocessor/processor.rs +114 -67
  184. package/rust/core/preprocessor/resolver/bank.rs +49 -47
  185. package/rust/core/preprocessor/resolver/call.rs +124 -53
  186. package/rust/core/preprocessor/resolver/condition.rs +95 -66
  187. package/rust/core/preprocessor/resolver/driver.rs +324 -182
  188. package/rust/core/preprocessor/resolver/function.rs +69 -0
  189. package/rust/core/preprocessor/resolver/group.rs +94 -118
  190. package/rust/core/preprocessor/resolver/let_.rs +32 -0
  191. package/rust/core/preprocessor/resolver/loop_.rs +318 -145
  192. package/rust/core/preprocessor/resolver/mod.rs +16 -10
  193. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  194. package/rust/core/preprocessor/resolver/spawn.rs +99 -53
  195. package/rust/core/preprocessor/resolver/synth.rs +54 -0
  196. package/rust/core/preprocessor/resolver/tempo.rs +48 -49
  197. package/rust/core/preprocessor/resolver/trigger.rs +116 -111
  198. package/rust/core/preprocessor/resolver/value.rs +176 -0
  199. package/rust/core/store/export.rs +28 -28
  200. package/rust/core/store/function.rs +40 -0
  201. package/rust/core/store/global.rs +61 -39
  202. package/rust/core/store/import.rs +28 -28
  203. package/rust/core/store/mod.rs +5 -4
  204. package/rust/core/store/variable.rs +51 -28
  205. package/rust/core/utils/mod.rs +1 -2
  206. package/rust/core/utils/path.rs +37 -46
  207. package/rust/lib.rs +308 -117
  208. package/rust/main.rs +364 -65
  209. package/rust/types/Cargo.toml +11 -0
  210. package/rust/types/src/addons.rs +55 -0
  211. package/rust/types/src/ast.rs +202 -0
  212. package/rust/types/src/config.rs +74 -0
  213. package/rust/types/src/lib.rs +12 -0
  214. package/rust/types/src/telemetry.rs +85 -0
  215. package/rust/utils/Cargo.toml +26 -0
  216. package/rust/utils/src/error.rs +186 -0
  217. package/rust/utils/src/file.rs +94 -0
  218. package/rust/utils/src/first_usage.rs +97 -0
  219. package/rust/utils/{mod.rs → src/lib.rs} +9 -6
  220. package/rust/utils/{logger.rs → src/logger.rs} +200 -123
  221. package/rust/utils/src/path.rs +88 -0
  222. package/rust/utils/src/signature.rs +41 -0
  223. package/rust/utils/{spinner.rs → src/spinner.rs} +20 -21
  224. package/rust/utils/src/version.rs +27 -0
  225. package/rust/utils/{watcher.rs → src/watcher.rs} +46 -33
  226. package/rust/web/api.rs +5 -0
  227. package/rust/web/cdn.rs +34 -0
  228. package/rust/web/mod.rs +3 -0
  229. package/rust/web/sso.rs +5 -0
  230. package/templates/minimal/README.md +143 -127
  231. package/templates/welcome/README.md +143 -127
  232. package/templates/welcome/src/index.deva +56 -8
  233. package/templates/welcome/src/variables.deva +2 -4
  234. package/tests/integration.rs +21 -0
  235. package/tests/rust/cli_check_build.rs +21 -0
  236. package/tests/rust/cli_help.rs +12 -0
  237. package/tests/rust/cli_template_list.rs +10 -0
  238. package/tests/rust/cli_version.rs +11 -0
  239. package/tests/typescript/index.spec.ts +136 -0
  240. package/tests/typescript/playhead.spec.ts +36 -0
  241. package/tests/typescript/render_e2e.spec.ts +77 -0
  242. package/tsconfig.json +12 -10
  243. package/typescript/bin/index.ts +19 -5
  244. package/typescript/core/functions/index.ts +83 -0
  245. package/typescript/core/index.ts +6 -0
  246. package/typescript/core/types/index.ts +4 -0
  247. package/typescript/core/types/plugin.ts +19 -0
  248. package/typescript/core/types/result.ts +29 -0
  249. package/typescript/core/types/statement.ts +47 -0
  250. package/typescript/core/types/value.ts +29 -0
  251. package/typescript/index.ts +8 -1
  252. package/typescript/pkg/devalang_core.d.ts +4 -0
  253. package/typescript/pkg/devalang_core.ts +49 -0
  254. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  255. package/typescript/scripts/postinstall.ts +85 -0
  256. package/typescript/scripts/version/bump.ts +0 -1
  257. package/typescript/scripts/version/fetch.ts +1 -6
  258. package/typescript/scripts/version/index.ts +0 -1
  259. package/docs/COMMANDS.md +0 -85
  260. package/docs/CONFIG.md +0 -30
  261. package/docs/SYNTAX.md +0 -210
  262. package/out-tsc/bin/devalang.exe +0 -0
  263. package/out-tsc/scripts/postbuild.js +0 -11
  264. package/rust/cli/build.rs +0 -137
  265. package/rust/cli/check.rs +0 -117
  266. package/rust/cli/play.rs +0 -193
  267. package/rust/config/loader.rs +0 -13
  268. package/rust/core/audio/engine.rs +0 -126
  269. package/rust/core/parser/handler/identifier.rs +0 -262
  270. package/rust/core/shared/duration.rs +0 -8
  271. package/rust/core/shared/mod.rs +0 -2
  272. package/rust/core/shared/value.rs +0 -18
  273. package/rust/core/utils/validation.rs +0 -35
  274. package/rust/utils/file.rs +0 -35
  275. package/rust/utils/signature.rs +0 -17
  276. package/rust/utils/version.rs +0 -15
  277. package/typescript/scripts/postbuild.ts +0 -8
@@ -0,0 +1,311 @@
1
+ use crate::core::{
2
+ audio::engine::AudioEngine,
3
+ parser::statement::{Statement, StatementKind},
4
+ plugin::runner::WasmPluginRunner,
5
+ store::{global::GlobalStore, variable::VariableTable},
6
+ };
7
+ use devalang_types::Value;
8
+ use devalang_utils::logger::{LogLevel, Logger};
9
+
10
+ use std::collections::HashMap;
11
+
12
+ pub fn interprete_call_arrow_statement(
13
+ stmt: &Statement,
14
+ audio_engine: &mut AudioEngine,
15
+ variable_table: &VariableTable,
16
+ global_store: &GlobalStore,
17
+ base_bpm: f32,
18
+ base_duration: f32,
19
+ max_end_time: &mut f32,
20
+ mut cursor_time: Option<&mut f32>,
21
+ update_cursor: bool,
22
+ ) -> (f32, f32) {
23
+ let cursor_copy = cursor_time.as_ref().map(|c| **c).unwrap_or(0.0);
24
+
25
+ if let StatementKind::ArrowCall {
26
+ target,
27
+ method,
28
+ args,
29
+ } = &stmt.kind
30
+ {
31
+ let Some(Value::Statement(synth_stmt)) = variable_table.get(target) else {
32
+ let logger = Logger::new();
33
+ logger.log_message(
34
+ LogLevel::Error,
35
+ &format!("Synth '{}' not found in variable table", target),
36
+ );
37
+ return (*max_end_time, cursor_copy);
38
+ };
39
+
40
+ let Value::Map(synth_map) = &synth_stmt.value else {
41
+ let logger = Logger::new();
42
+ logger.log_message(
43
+ LogLevel::Error,
44
+ &format!("Invalid synth statement for '{}', expected a map.", target),
45
+ );
46
+ return (*max_end_time, cursor_copy);
47
+ };
48
+
49
+ let Some(Value::String(entity)) = synth_map.get("entity") else {
50
+ let logger = Logger::new();
51
+ logger.log_message(
52
+ LogLevel::Error,
53
+ &format!("Missing 'entity' key in synth '{}'.", target),
54
+ );
55
+ return (*max_end_time, cursor_copy);
56
+ };
57
+
58
+ if entity != "synth" {
59
+ let logger = Logger::new();
60
+ logger.log_message(
61
+ LogLevel::Error,
62
+ &format!("'{}' is not a synth, entity is '{}'.", target, entity),
63
+ );
64
+ return (*max_end_time, cursor_copy);
65
+ }
66
+
67
+ let Some(Value::Map(value_map)) = synth_map.get("value") else {
68
+ let logger = Logger::new();
69
+ logger.log_message(
70
+ LogLevel::Error,
71
+ &format!("Missing 'value' map in synth '{}'.", target),
72
+ );
73
+ return (*max_end_time, cursor_copy);
74
+ };
75
+
76
+ let waveform_str = match value_map.get("waveform") {
77
+ Some(Value::String(s)) => s.clone(),
78
+ Some(Value::Identifier(s)) => s.clone(),
79
+ _ => {
80
+ let logger = Logger::new();
81
+ logger.log_message(
82
+ LogLevel::Error,
83
+ &format!("Missing or invalid 'waveform' in synth '{}'.", target),
84
+ );
85
+ return (*max_end_time, cursor_copy);
86
+ }
87
+ };
88
+ let Some(Value::Map(params)) = value_map.get("parameters") else {
89
+ println!("❌ Missing or invalid 'parameters' in synth '{}'.", target);
90
+ return (*max_end_time, cursor_copy);
91
+ };
92
+
93
+ // Synth parameters
94
+ let synth_params = params.clone();
95
+ let amp = extract_f32(&synth_params, "amp", base_bpm).unwrap_or(1.0);
96
+
97
+ if method == "note" {
98
+ let filtered_args: Vec<_> = args
99
+ .iter()
100
+ .filter(|arg| !matches!(arg, Value::Unknown))
101
+ .collect();
102
+
103
+ let Some(Value::Identifier(note_name)) = filtered_args.first().map(|v| (*v).clone())
104
+ else {
105
+ println!(
106
+ "❌ Invalid or missing argument for 'note' method on '{}'.",
107
+ target
108
+ );
109
+ return (*max_end_time, cursor_copy);
110
+ };
111
+
112
+ let mut note_params = HashMap::new();
113
+ if let Some(arg1) = filtered_args.get(1) {
114
+ if let Value::Map(map) = (*arg1).clone() {
115
+ for (key, value) in map {
116
+ note_params.insert(key, value);
117
+ }
118
+ }
119
+ }
120
+
121
+ // Note parameters and calculations
122
+ let amp_note = extract_f32(&note_params, "amp", base_bpm).unwrap_or(amp);
123
+ let duration_ms =
124
+ extract_f32(&note_params, "duration", base_bpm).unwrap_or(base_duration * 1000.0);
125
+
126
+ let duration_secs = duration_ms / 1000.0;
127
+ let final_freq = note_to_freq(&note_name);
128
+ let start_time = cursor_copy;
129
+ let end_time = start_time + duration_secs;
130
+
131
+ // Fetch automation map if present:
132
+ // - Global (per-synth): key "<target>__automation" => map with key "params"
133
+ // - Per-note: note parameter "automate" => map
134
+ let auto_key = format!("{}__automation", target);
135
+ let synth_automation = match variable_table.get(&auto_key) {
136
+ Some(Value::Map(map)) => match map.get("params") {
137
+ Some(Value::Map(p)) => Some(p.clone()),
138
+ _ => None,
139
+ },
140
+ _ => None,
141
+ };
142
+
143
+ let note_automation = match note_params.get("automate") {
144
+ Some(Value::Map(m)) => Some(m.clone()),
145
+ _ => None,
146
+ };
147
+
148
+ // Merge: per-note overrides synth automation per key (volume/pan/pitch)
149
+ let automation = match (synth_automation, note_automation) {
150
+ (Some(mut a), Some(n)) => {
151
+ for (k, v) in n {
152
+ a.insert(k, v);
153
+ }
154
+ Some(a)
155
+ }
156
+ (None, Some(n)) => Some(n),
157
+ (Some(a), None) => Some(a),
158
+ _ => None,
159
+ };
160
+
161
+ // If waveform references a plugin alias (e.g., alias.synth), use the WASM plugin runner
162
+ if waveform_str.contains('.') && waveform_str.ends_with(".synth") {
163
+ let alias = waveform_str.split('.').next().unwrap_or("");
164
+ if let Some(Value::String(uri)) = variable_table.get(alias) {
165
+ if let Some(id) = uri.strip_prefix("devalang://plugin/") {
166
+ let mut parts = id.split('.');
167
+ let author = parts.next().unwrap_or("");
168
+ let name = parts.next().unwrap_or("");
169
+ let key = format!("{}:{}", author, name);
170
+ if let Some((_info, wasm_bytes)) = global_store.plugins.get(&key) {
171
+ // Prepare buffer (stereo f32)
172
+ let sample_rate = 44100.0_f32;
173
+ let total_samples = ((duration_ms / 1000.0) * sample_rate) as usize;
174
+ let channels = 2usize;
175
+ let start_index = ((start_time * sample_rate) as usize) * channels;
176
+ let required_len = start_index + total_samples * channels;
177
+ if audio_engine.buffer.len() < required_len {
178
+ audio_engine.buffer.resize(required_len, 0);
179
+ }
180
+ let mut fbuf = vec![0.0f32; total_samples * channels];
181
+ let runner = WasmPluginRunner::new();
182
+ let mut params_num: std::collections::HashMap<String, f32> =
183
+ std::collections::HashMap::new();
184
+ let mut params_str: std::collections::HashMap<String, String> =
185
+ std::collections::HashMap::new();
186
+ for (k, v) in synth_params.iter() {
187
+ match v {
188
+ Value::Number(n) => {
189
+ params_num.insert(k.clone(), *n);
190
+ }
191
+ Value::String(s) => {
192
+ params_str.insert(k.clone(), s.clone());
193
+ }
194
+ Value::Identifier(s) => {
195
+ params_str.insert(k.clone(), s.clone());
196
+ }
197
+ _ => {}
198
+ }
199
+ }
200
+ let _ = runner.render_note_with_params_in_place(
201
+ wasm_bytes,
202
+ &mut fbuf,
203
+ None,
204
+ final_freq,
205
+ amp_note,
206
+ duration_ms as i32,
207
+ 44100,
208
+ 2,
209
+ &params_num,
210
+ Some(&params_str),
211
+ );
212
+ for (i, sample) in
213
+ fbuf.iter().enumerate().take(total_samples * channels)
214
+ {
215
+ let s = (sample.clamp(-1.0, 1.0) * (i16::MAX as f32)) as i16;
216
+ let idx = start_index + i;
217
+ audio_engine.buffer[idx] =
218
+ audio_engine.buffer[idx].saturating_add(s);
219
+ }
220
+ } else {
221
+ let logger = Logger::new();
222
+ logger.log_message(
223
+ LogLevel::Warning,
224
+ &format!(
225
+ "Plugin bytes not found for key '{}' (alias '{}').",
226
+ key, alias
227
+ ),
228
+ );
229
+ }
230
+ } else {
231
+ let logger = Logger::new();
232
+ logger.log_message(
233
+ LogLevel::Warning,
234
+ &format!("Invalid plugin URI in alias '{}': {}", alias, uri),
235
+ );
236
+ }
237
+ } else {
238
+ let logger = Logger::new();
239
+ logger.log_message(
240
+ LogLevel::Warning,
241
+ &format!("Plugin alias '{}' not found in variable table.", alias),
242
+ );
243
+ }
244
+ } else {
245
+ audio_engine.insert_note(
246
+ waveform_str.clone(),
247
+ final_freq,
248
+ amp_note,
249
+ start_time * 1000.0,
250
+ duration_ms,
251
+ synth_params,
252
+ note_params,
253
+ automation,
254
+ );
255
+ }
256
+
257
+ *max_end_time = (*max_end_time).max(end_time);
258
+
259
+ if update_cursor {
260
+ if let Some(c) = cursor_time.as_mut() {
261
+ **c = end_time;
262
+ }
263
+ }
264
+
265
+ return (*max_end_time, end_time);
266
+ } else {
267
+ let logger = Logger::new();
268
+ logger.log_message(
269
+ LogLevel::Error,
270
+ &format!("Unknown method '{}' on synth '{}'.", method, target),
271
+ );
272
+ }
273
+ }
274
+
275
+ (*max_end_time, cursor_copy)
276
+ }
277
+
278
+ fn extract_f32(map: &HashMap<String, Value>, key: &str, base_bpm: f32) -> Option<f32> {
279
+ map.get(key).and_then(|v| match v {
280
+ Value::Number(n) => Some(*n),
281
+ Value::Beat(beat_str) => {
282
+ let parts: Vec<&str> = beat_str.split('/').collect();
283
+ if parts.len() == 2 {
284
+ let numerator = parts[0].parse::<f32>().ok()?;
285
+ let denominator = parts[1].parse::<f32>().ok()?;
286
+
287
+ Some((numerator / denominator) * ((60.0 / base_bpm) * 1000.0))
288
+ } else {
289
+ None
290
+ }
291
+ }
292
+ _ => None,
293
+ })
294
+ }
295
+
296
+ fn note_to_freq(note: &str) -> f32 {
297
+ let notes = [
298
+ "C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B",
299
+ ];
300
+
301
+ if note.len() < 2 || note.len() > 3 {
302
+ return 440.0;
303
+ }
304
+
305
+ let (name, octave_str) = note.split_at(note.len() - 1);
306
+ let semitone = notes.iter().position(|&n| n == name).unwrap_or(9) as i32;
307
+ let octave = octave_str.parse::<i32>().unwrap_or(4);
308
+ let midi_note = (octave + 1) * 12 + semitone;
309
+
310
+ 440.0 * (2.0_f32).powf(((midi_note as f32) - 69.0) / 12.0)
311
+ }
@@ -0,0 +1,18 @@
1
+ use crate::core::{
2
+ parser::statement::{Statement, StatementKind},
3
+ store::variable::VariableTable,
4
+ };
5
+
6
+ // Store automation configuration into the variable table under a namespaced key
7
+ // Key: "<target>__automation" => Value::Map({ target, params })
8
+ pub fn interprete_automate_statement(
9
+ stmt: &Statement,
10
+ variable_table: &mut VariableTable,
11
+ ) -> Option<VariableTable> {
12
+ if let StatementKind::Automate { target } = &stmt.kind {
13
+ let key = format!("{}__automation", target);
14
+ variable_table.set(key, stmt.value.clone());
15
+ return Some(variable_table.clone());
16
+ }
17
+ None
18
+ }
@@ -1,42 +1,294 @@
1
- use crate::core::{
2
- audio::{ engine::AudioEngine, interpreter::{ driver::execute_audio_block } },
3
- parser::statement::{ Statement, StatementKind },
4
- shared::{ duration::Duration, value::Value },
5
- store::variable::VariableTable,
6
- };
7
-
8
- pub fn interprete_call_statement(
9
- stmt: &Statement,
10
- audio_engine: AudioEngine,
11
- variable_table: VariableTable,
12
- base_bpm: f32,
13
- base_duration: f32,
14
- max_end_time: f32,
15
- cursor_time: f32
16
- ) -> (AudioEngine, f32, f32) {
17
- if let Value::String(identifier) = &stmt.value {
18
- if let Some(Value::Map(map)) = variable_table.clone().get(identifier) {
19
- if let Some(Value::Block(block)) = map.get("body") {
20
- let (eng, _, end_time) = execute_audio_block(
21
- audio_engine,
22
- variable_table,
23
- block.clone(),
24
- base_bpm,
25
- base_duration,
26
- max_end_time,
27
- cursor_time
28
- );
29
-
30
- return (eng, max_end_time.max(end_time), end_time);
31
- } else {
32
- eprintln!("❌ 'body' must be a block");
33
- }
34
- } else {
35
- eprintln!("❌ Call target '{}' not found or invalid", identifier);
36
- }
37
- } else {
38
- eprintln!("❌ Invalid call statement: expected string identifier");
39
- }
40
-
41
- (audio_engine, max_end_time, cursor_time)
42
- }
1
+ use devalang_types::{Duration, Value};
2
+
3
+ use crate::core::{
4
+ audio::{engine::AudioEngine, interpreter::driver::execute_audio_block},
5
+ parser::statement::{Statement, StatementKind},
6
+ store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
7
+ };
8
+
9
+ pub fn interprete_call_statement(
10
+ stmt: &Statement,
11
+ audio_engine: &mut AudioEngine,
12
+ variable_table: &VariableTable,
13
+ functions: &FunctionTable,
14
+ global_store: &GlobalStore,
15
+ base_bpm: f32,
16
+ base_duration: f32,
17
+ max_end_time: f32,
18
+ cursor_time: f32,
19
+ ) -> (f32, f32) {
20
+ if let StatementKind::Call { name, args } = &stmt.kind {
21
+ // Classic function call case
22
+ if let Some(func) = functions.functions.get(name) {
23
+ // function found
24
+ if func.parameters.len() != args.len() {
25
+ eprintln!(
26
+ "❌ Function '{}' expects {} args, got {}",
27
+ name,
28
+ func.parameters.len(),
29
+ args.len()
30
+ );
31
+ return (max_end_time, cursor_time);
32
+ }
33
+
34
+ let mut local_vars = VariableTable::with_parent(variable_table.clone());
35
+ for (param, arg) in func.parameters.iter().zip(args) {
36
+ local_vars.set(param.clone(), arg.clone());
37
+ }
38
+
39
+ return execute_audio_block(
40
+ audio_engine,
41
+ global_store,
42
+ local_vars,
43
+ functions.clone(),
44
+ &func.body,
45
+ base_bpm,
46
+ base_duration,
47
+ max_end_time,
48
+ cursor_time,
49
+ );
50
+ }
51
+
52
+ // Group case
53
+ if let Some(group_stmt) = find_group(name, variable_table, global_store) {
54
+ // group found
55
+ if let Value::Map(map) = &group_stmt.value {
56
+ if let Some(Value::Block(body)) = map.get("body") {
57
+ return execute_audio_block(
58
+ audio_engine,
59
+ global_store,
60
+ variable_table.clone(),
61
+ functions.clone(),
62
+ body,
63
+ base_bpm,
64
+ base_duration,
65
+ max_end_time,
66
+ cursor_time,
67
+ );
68
+ }
69
+ }
70
+ }
71
+
72
+ // Pattern case
73
+ if let Some(pattern_stmt) = find_pattern(name, variable_table, global_store) {
74
+ // Extract pattern string from statement value
75
+ if let Value::String(pat) = &pattern_stmt.value {
76
+ // Determine target entity (explicit or inferred)
77
+ let mut target_entity = name.clone();
78
+ if let StatementKind::Pattern { name: _n, target } = &pattern_stmt.kind {
79
+ if let Some(t) = target {
80
+ target_entity = t.clone();
81
+ }
82
+ }
83
+
84
+ // Build a variable table snapshot for resolution like triggers do
85
+ // Preserve the full parent chain so lookups behave the same as runtime
86
+ fn clone_with_parents(
87
+ orig: &crate::core::store::variable::VariableTable,
88
+ ) -> crate::core::store::variable::VariableTable {
89
+ crate::core::store::variable::VariableTable {
90
+ variables: orig.variables.clone(),
91
+ parent: orig
92
+ .parent
93
+ .as_ref()
94
+ .map(|p| Box::new(clone_with_parents(p))),
95
+ }
96
+ }
97
+
98
+ let final_variable_table = clone_with_parents(variable_table);
99
+
100
+ // Normalize pattern: remove spaces and line breaks
101
+ let pattern_str: String = pat.chars().filter(|c| !c.is_whitespace()).collect();
102
+ if pattern_str.is_empty() {
103
+ return (max_end_time, cursor_time);
104
+ }
105
+
106
+ let step_count = pattern_str.len() as f32;
107
+ // Assume pattern spans one bar (4 beats)
108
+ let total_bar = 4.0 * base_duration;
109
+ let step_duration = total_bar / step_count; // seconds per step
110
+
111
+ let mut updated_max = max_end_time;
112
+
113
+ for (i, ch) in pattern_str.chars().enumerate() {
114
+ if ch == '-' {
115
+ continue; // rest
116
+ }
117
+
118
+ // Schedule a trigger at cursor_time + offset
119
+ let event_time = cursor_time + (i as f32) * step_duration;
120
+
121
+ // Resolve trigger value similarly to interprete_trigger_statement
122
+ let mut trigger_val = Value::String(target_entity.clone());
123
+ if let Some(val) = variable_table.variables.get(&target_entity) {
124
+ match val {
125
+ Value::Identifier(id) => {
126
+ // resolve from parent if available
127
+ if let Some(parent) = &variable_table.parent {
128
+ if let Some(v) = parent.get(id) {
129
+ trigger_val = v.clone();
130
+ }
131
+ } else if let Some(v) = variable_table.get(id) {
132
+ trigger_val = v.clone();
133
+ }
134
+ }
135
+ Value::Map(map) => {
136
+ if let Some(Value::String(src)) = map.get("entity") {
137
+ trigger_val = Value::String(src.clone());
138
+ } else if let Some(Value::Identifier(src)) = map.get("entity") {
139
+ trigger_val = Value::Identifier(src.clone());
140
+ }
141
+ }
142
+ Value::Sample(sample_src) => {
143
+ trigger_val = Value::Sample(sample_src.clone());
144
+ }
145
+ _ => {
146
+ // leave as string
147
+ }
148
+ }
149
+ }
150
+
151
+ // Use loader to get sample path and sample length
152
+ let (src, sample_length) = crate::core::audio::loader::trigger::load_trigger(
153
+ &trigger_val,
154
+ &Duration::Number(step_duration),
155
+ &None,
156
+ base_duration,
157
+ final_variable_table.clone(),
158
+ );
159
+
160
+ let play_length = step_duration.min(sample_length);
161
+
162
+ let trigger_src = match trigger_val.get("entity") {
163
+ Some(Value::String(s)) => s.clone(),
164
+ Some(Value::Identifier(id)) => id.clone(),
165
+ Some(Value::Statement(stmt)) => {
166
+ if let StatementKind::Trigger { entity, .. } = &stmt.kind {
167
+ entity.clone()
168
+ } else {
169
+ src.clone()
170
+ }
171
+ }
172
+ _ => src.clone(),
173
+ };
174
+
175
+ audio_engine.insert_sample(
176
+ &trigger_src,
177
+ event_time,
178
+ play_length,
179
+ None,
180
+ &final_variable_table,
181
+ );
182
+
183
+ let end_time = event_time + play_length;
184
+ if end_time > updated_max {
185
+ updated_max = end_time;
186
+ }
187
+ }
188
+
189
+ return (updated_max, cursor_time);
190
+ }
191
+ }
192
+
193
+ // Function or group not found; keep as debug-free fail path
194
+ }
195
+
196
+ (max_end_time, cursor_time)
197
+ }
198
+
199
+ fn find_group(
200
+ name: &str,
201
+ variable_table: &VariableTable,
202
+ global_store: &GlobalStore,
203
+ ) -> Option<Statement> {
204
+ use crate::core::parser::statement::Statement;
205
+ use crate::core::parser::statement::StatementKind;
206
+
207
+ if let Some(Value::Statement(stmt_box)) = variable_table.get(name) {
208
+ if let StatementKind::Group = stmt_box.kind {
209
+ return Some(*stmt_box.clone());
210
+ }
211
+ }
212
+
213
+ if let Some(val) = global_store.variables.variables.get(name) {
214
+ match val {
215
+ Value::Statement(stmt_box) => {
216
+ if let StatementKind::Group = stmt_box.kind {
217
+ return Some(*stmt_box.clone());
218
+ }
219
+ }
220
+ Value::Map(map) => {
221
+ // Try to rebuild a Group statement from the stored map
222
+ if let (Some(Value::String(_id)), Some(Value::Block(_body))) =
223
+ (map.get("identifier"), map.get("body"))
224
+ {
225
+ let stmt = Statement {
226
+ kind: StatementKind::Group,
227
+ value: Value::Map(map.clone()),
228
+ indent: 0,
229
+ line: 0,
230
+ column: 0,
231
+ };
232
+ return Some(stmt);
233
+ }
234
+ }
235
+ _ => {}
236
+ }
237
+ }
238
+
239
+ None
240
+ }
241
+
242
+ fn find_pattern(
243
+ name: &str,
244
+ variable_table: &VariableTable,
245
+ global_store: &GlobalStore,
246
+ ) -> Option<Statement> {
247
+ use crate::core::parser::statement::Statement;
248
+ use crate::core::parser::statement::StatementKind;
249
+
250
+ if let Some(Value::Statement(stmt_box)) = variable_table.get(name) {
251
+ if let StatementKind::Pattern { .. } = stmt_box.kind {
252
+ return Some(*stmt_box.clone());
253
+ }
254
+ }
255
+
256
+ if let Some(val) = global_store.variables.variables.get(name) {
257
+ match val {
258
+ Value::Statement(stmt_box) => {
259
+ if let StatementKind::Pattern { .. } = stmt_box.kind {
260
+ return Some(*stmt_box.clone());
261
+ }
262
+ }
263
+ Value::Map(map) => {
264
+ if let Some(Value::String(_pat)) = map.get("pattern") {
265
+ // Rebuild a Pattern statement from stored map if possible
266
+ let stmt = Statement {
267
+ kind: StatementKind::Pattern {
268
+ name: name.to_string(),
269
+ target: map.get("target").and_then(|v| match v {
270
+ Value::String(s) => Some(s.clone()),
271
+ _ => None,
272
+ }),
273
+ },
274
+ value: Value::String(
275
+ map.get("pattern")
276
+ .and_then(|v| match v {
277
+ Value::String(s) => Some(s.clone()),
278
+ _ => None,
279
+ })
280
+ .unwrap_or_default(),
281
+ ),
282
+ indent: 0,
283
+ line: 0,
284
+ column: 0,
285
+ };
286
+ return Some(stmt);
287
+ }
288
+ }
289
+ _ => {}
290
+ }
291
+ }
292
+
293
+ None
294
+ }