@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
@@ -1,310 +1,152 @@
1
- use crate::core::audio::special::{
2
- find_and_eval_first_easing_call, find_and_eval_first_math_call, find_and_eval_first_mod_call,
3
- resolve_env_atom,
4
- };
5
- use crate::core::store::variable::VariableTable;
6
- use devalang_types::Value;
7
-
8
- pub fn evaluate_condition_string(expr: &str, vars: &VariableTable) -> bool {
9
- let tokens: Vec<&str> = expr.split_whitespace().collect();
10
- if tokens.len() != 3 {
11
- return false;
12
- }
13
-
14
- let left = tokens[0];
15
- let op = tokens[1];
16
- let right = tokens[2];
17
-
18
- // Resolve left and right to numeric values where possible. Accept numbers, variables or env atoms.
19
- fn resolve_for_cond(s: &str, vars: &VariableTable) -> Option<f32> {
20
- if let Ok(n) = s.parse::<f32>() {
21
- return Some(n);
22
- }
23
- if let Some(Value::Number(n)) = vars.get(s) {
24
- return Some(*n);
25
- }
26
- if let Some(v) = resolve_env_atom(s, 120.0, 1.0) {
27
- return Some(v);
28
- }
29
- None
30
- }
31
-
32
- let left_val = match resolve_for_cond(left, vars) {
33
- Some(v) => v,
34
- None => return false,
35
- };
36
-
37
- let right_val = match resolve_for_cond(right, vars) {
38
- Some(v) => v,
39
- None => return false,
40
- };
41
-
42
- match op {
43
- ">" => left_val > right_val,
44
- "<" => left_val < right_val,
45
- ">=" => left_val >= right_val,
46
- "<=" => left_val <= right_val,
47
- "==" => {
48
- // relative epsilon for floating comparisons
49
- let diff = (left_val - right_val).abs();
50
- let largest = left_val.abs().max(right_val.abs()).max(1.0);
51
- diff <= (f32::EPSILON * largest)
52
- }
53
- "!=" => {
54
- let diff = (left_val - right_val).abs();
55
- let largest = left_val.abs().max(right_val.abs()).max(1.0);
56
- diff > (f32::EPSILON * largest)
57
- }
58
- _ => false,
59
- }
60
- }
61
-
62
- // Very small expression evaluator for `$env.*`, `$math.*` and variables.
63
- // Supports: +, -, *, / and simple parentheses, left-to-right (no precedence), and $math.(sin|cos)(expr)
64
- pub fn evaluate_numeric_expression(
65
- expr: &str,
66
- vars: &VariableTable,
67
- env_bpm: f32,
68
- env_beat: f32,
69
- ) -> Option<f32> {
70
- let expr = expr.replace(" ", "");
71
-
72
- // Helper to resolve an atom to a number
73
- fn resolve_atom(atom: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
74
- if let Some(v) = resolve_env_atom(atom, bpm, beat) {
75
- return Some(v);
76
- }
77
- if let Ok(n) = atom.parse::<f32>() {
78
- return Some(n);
79
- }
80
- if let Some(Value::Number(n)) = vars.get(atom) {
81
- return Some(*n);
82
- }
83
- None
84
- }
85
-
86
- // Shunting-like, simplified: first evaluate any $math.func(...) calls anywhere in the expression,
87
- // then fold remaining parentheses and evaluate left-to-right.
88
- fn eval(expr: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
89
- // 1) Replace $math/$easing/$mod calls progressively with a max iteration guard
90
- let mut s = expr.to_string();
91
- let mut iterations = 0u32;
92
- const MAX_ITER: u32 = 64;
93
-
94
- // Evaluate modulators first (they may feed easing/math)
95
- while iterations < MAX_ITER {
96
- if let Some(next) =
97
- find_and_eval_first_mod_call(&s, evaluate_numeric_expression, vars, bpm, beat)
98
- {
99
- s = next;
100
- iterations += 1;
101
- continue;
102
- }
103
- break;
104
- }
105
-
106
- iterations = 0;
107
- while iterations < MAX_ITER {
108
- if let Some(next) =
109
- find_and_eval_first_easing_call(&s, evaluate_numeric_expression, vars, bpm, beat)
110
- {
111
- s = next;
112
- iterations += 1;
113
- continue;
114
- }
115
- break;
116
- }
117
-
118
- iterations = 0;
119
- while iterations < MAX_ITER {
120
- if let Some(next) =
121
- find_and_eval_first_math_call(&s, evaluate_numeric_expression, vars, bpm, beat)
122
- {
123
- s = next;
124
- iterations += 1;
125
- continue;
126
- }
127
- break;
128
- }
129
-
130
- // 2) Evaluate remaining (pure) parentheses starting from innermost
131
- if let Some(open) = s.rfind('(') {
132
- if let Some(close_rel) = s[open..].find(')') {
133
- // index relatif
134
- let close = open + close_rel;
135
- let inner = &s[open + 1..close];
136
- let val = eval(inner, vars, bpm, beat)?;
137
- let mut replaced = String::new();
138
- replaced.push_str(&s[..open]);
139
- replaced.push_str(&val.to_string());
140
- replaced.push_str(&s[close + 1..]);
141
- return eval(&replaced, vars, bpm, beat);
142
- }
143
- }
144
-
145
- // Tokenize by operators left-to-right
146
- let mut parts: Vec<String> = Vec::new();
147
- let mut cur = String::new();
148
- for ch in s.chars() {
149
- if "+-*/".contains(ch) {
150
- if !cur.is_empty() {
151
- parts.push(cur.clone());
152
- cur.clear();
153
- }
154
- parts.push(ch.to_string());
155
- } else {
156
- cur.push(ch);
157
- }
158
- }
159
- if !cur.is_empty() {
160
- parts.push(cur);
161
- }
162
- if parts.is_empty() {
163
- return None;
164
- }
165
-
166
- // Resolve atoms and compute
167
- let mut acc: Option<f32> = None;
168
- let mut op: Option<char> = None;
169
- for part in parts {
170
- if part.len() == 1 && "+-*/".contains(part.chars().next().unwrap()) {
171
- op = part.chars().next();
172
- continue;
173
- }
174
- let val = if let Some(v) = resolve_atom(&part, vars, bpm, beat) {
175
- v
176
- } else if part.starts_with("$env.") {
177
- // $env atom not handled by resolve_atom (when composed), try recursive eval
178
- eval(&part, vars, bpm, beat)?
179
- } else {
180
- return None;
181
- };
182
-
183
- acc = Some(match (acc, op) {
184
- (None, _) => val,
185
- (Some(a), Some('+')) => a + val,
186
- (Some(a), Some('-')) => a - val,
187
- (Some(a), Some('*')) => a * val,
188
- (Some(a), Some('/')) => {
189
- if val != 0.0 {
190
- a / val
191
- } else {
192
- return Some(f32::INFINITY);
193
- }
194
- }
195
- (Some(_), None) => val,
196
- _ => {
197
- return None;
198
- }
199
- });
200
- }
201
-
202
- acc
203
- }
204
-
205
- eval(&expr, vars, env_bpm, env_beat)
206
- }
207
-
208
- pub fn evaluate_rhs_into_value(
209
- raw: &str,
210
- vars: &VariableTable,
211
- env_bpm: f32,
212
- env_beat: f32,
213
- ) -> Value {
214
- if let Some(num) = evaluate_numeric_expression(raw, vars, env_bpm, env_beat) {
215
- Value::Number(num)
216
- } else {
217
- Value::String(raw.to_string())
218
- }
219
- }
220
-
221
- // Evaluate a simple string concatenation expression like: "hello " + name + "!" + $env.beat
222
- // - Splits on + outside quotes
223
- // - Terms can be string literals (double quotes), variables (Number/String/Boolean), or numeric env/math expressions
224
- // Returns None if parsing fails (fallback to raw print)
225
- pub fn evaluate_string_expression(
226
- expr: &str,
227
- vars: &VariableTable,
228
- env_bpm: f32,
229
- env_beat: f32,
230
- ) -> Option<String> {
231
- // Quick reject if no '+' present
232
- if !expr.contains('+') {
233
- return None;
234
- }
235
-
236
- // Split by '+' outside of quotes
237
- let mut parts: Vec<String> = Vec::new();
238
- let mut cur = String::new();
239
- let mut in_quotes = false;
240
- let mut escape = false;
241
- for ch in expr.chars() {
242
- if escape {
243
- cur.push(ch);
244
- escape = false;
245
- continue;
246
- }
247
- if ch == '\\' {
248
- // escape next char
249
- escape = true;
250
- continue;
251
- }
252
- if ch == '"' {
253
- in_quotes = !in_quotes;
254
- cur.push(ch);
255
- continue;
256
- }
257
- if ch == '+' && !in_quotes {
258
- parts.push(cur.to_string());
259
- cur.clear();
260
- continue;
261
- }
262
- cur.push(ch);
263
- }
264
- if !cur.is_empty() {
265
- parts.push(cur.to_string());
266
- }
267
- if parts.is_empty() {
268
- return None;
269
- }
270
-
271
- // Resolve each part into a string
272
- fn strip_quotes(s: &str) -> Option<String> {
273
- let st = s.trim();
274
- if st.len() >= 2 && st.starts_with('"') && st.ends_with('"') {
275
- Some(st[1..st.len() - 1].to_string())
276
- } else {
277
- None
278
- }
279
- }
280
-
281
- let mut out = String::new();
282
- for p in parts {
283
- if p.is_empty() {
284
- continue;
285
- }
286
- if let Some(lit) = strip_quotes(&p) {
287
- out.push_str(&lit);
288
- continue;
289
- }
290
- // Try variables first
291
- if let Some(val) = vars.get(&p) {
292
- match val {
293
- Value::String(s) => out.push_str(s),
294
- Value::Number(n) => out.push_str(&format!("{}", n)),
295
- Value::Boolean(b) => out.push_str(&format!("{}", b)),
296
- other => out.push_str(&format!("{:?}", other)),
297
- }
298
- continue;
299
- }
300
- // Try env/math/numeric expression for this term
301
- if let Some(n) = evaluate_numeric_expression(&p, vars, env_bpm, env_beat) {
302
- out.push_str(&format!("{}", n));
303
- continue;
304
- }
305
- // Bareword not resolved: include as-is (safe fallback)
306
- out.push_str(&p);
307
- }
308
-
309
- Some(out)
310
- }
1
+ use crate::core::audio::special::{
2
+ find_and_eval_first_easing_call, find_and_eval_first_math_call, find_and_eval_first_mod_call,
3
+ resolve_env_atom,
4
+ };
5
+ use devalang_types::Value;
6
+ use devalang_types::VariableTable;
7
+
8
+ // Very small expression evaluator for `$env.*`, `$math.*` and variables.
9
+ // Supports: +, -, *, / and simple parentheses, left-to-right (no precedence), and $math.(sin|cos)(expr)
10
+ pub fn evaluate_numeric_expression(
11
+ expr: &str,
12
+ vars: &VariableTable,
13
+ env_bpm: f32,
14
+ env_beat: f32,
15
+ ) -> Option<f32> {
16
+ let expr = expr.replace(" ", "");
17
+
18
+ // Helper to resolve an atom to a number
19
+ fn resolve_atom(atom: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
20
+ if let Some(v) = resolve_env_atom(atom, bpm, beat) {
21
+ return Some(v);
22
+ }
23
+ if let Ok(n) = atom.parse::<f32>() {
24
+ return Some(n);
25
+ }
26
+ if let Some(Value::Number(n)) = vars.get(atom) {
27
+ return Some(*n);
28
+ }
29
+ None
30
+ }
31
+
32
+ // Shunting-like, simplified: first evaluate any $math.func(...) calls anywhere in the expression,
33
+ // then fold remaining parentheses and evaluate left-to-right.
34
+ fn eval(expr: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
35
+ // 1) Replace $math/$easing/$mod calls progressively with a max iteration guard
36
+ let mut s = expr.to_string();
37
+ let mut iterations = 0u32;
38
+ const MAX_ITER: u32 = 64;
39
+
40
+ // Evaluate modulators first (they may feed easing/math)
41
+ while iterations < MAX_ITER {
42
+ if let Some(next) =
43
+ find_and_eval_first_mod_call(&s, evaluate_numeric_expression, vars, bpm, beat)
44
+ {
45
+ s = next;
46
+ iterations += 1;
47
+ continue;
48
+ }
49
+ break;
50
+ }
51
+
52
+ iterations = 0;
53
+ while iterations < MAX_ITER {
54
+ if let Some(next) =
55
+ find_and_eval_first_easing_call(&s, evaluate_numeric_expression, vars, bpm, beat)
56
+ {
57
+ s = next;
58
+ iterations += 1;
59
+ continue;
60
+ }
61
+ break;
62
+ }
63
+
64
+ iterations = 0;
65
+ while iterations < MAX_ITER {
66
+ if let Some(next) =
67
+ find_and_eval_first_math_call(&s, evaluate_numeric_expression, vars, bpm, beat)
68
+ {
69
+ s = next;
70
+ iterations += 1;
71
+ continue;
72
+ }
73
+ break;
74
+ }
75
+
76
+ // 2) Evaluate remaining (pure) parentheses starting from innermost
77
+ if let Some(open) = s.rfind('(') {
78
+ if let Some(close_rel) = s[open..].find(')') {
79
+ // index relatif
80
+ let close = open + close_rel;
81
+ let inner = &s[open + 1..close];
82
+ let val = eval(inner, vars, bpm, beat)?;
83
+ let mut replaced = String::new();
84
+ replaced.push_str(&s[..open]);
85
+ replaced.push_str(&val.to_string());
86
+ replaced.push_str(&s[close + 1..]);
87
+ return eval(&replaced, vars, bpm, beat);
88
+ }
89
+ }
90
+
91
+ // Tokenize by operators left-to-right
92
+ let mut parts: Vec<String> = Vec::new();
93
+ let mut cur = String::new();
94
+ for ch in s.chars() {
95
+ if "+-*/".contains(ch) {
96
+ if !cur.is_empty() {
97
+ parts.push(cur.clone());
98
+ cur.clear();
99
+ }
100
+ parts.push(ch.to_string());
101
+ } else {
102
+ cur.push(ch);
103
+ }
104
+ }
105
+ if !cur.is_empty() {
106
+ parts.push(cur);
107
+ }
108
+ if parts.is_empty() {
109
+ return None;
110
+ }
111
+
112
+ // Resolve atoms and compute
113
+ let mut acc: Option<f32> = None;
114
+ let mut op: Option<char> = None;
115
+ for part in parts {
116
+ if part.len() == 1 && "+-*/".contains(part.chars().next().unwrap()) {
117
+ op = part.chars().next();
118
+ continue;
119
+ }
120
+ let val = if let Some(v) = resolve_atom(&part, vars, bpm, beat) {
121
+ v
122
+ } else if part.starts_with("$env.") {
123
+ // $env atom not handled by resolve_atom (when composed), try recursive eval
124
+ eval(&part, vars, bpm, beat)?
125
+ } else {
126
+ return None;
127
+ };
128
+
129
+ acc = Some(match (acc, op) {
130
+ (None, _) => val,
131
+ (Some(a), Some('+')) => a + val,
132
+ (Some(a), Some('-')) => a - val,
133
+ (Some(a), Some('*')) => a * val,
134
+ (Some(a), Some('/')) => {
135
+ if val != 0.0 {
136
+ a / val
137
+ } else {
138
+ return Some(f32::INFINITY);
139
+ }
140
+ }
141
+ (Some(_), None) => val,
142
+ _ => {
143
+ return None;
144
+ }
145
+ });
146
+ }
147
+
148
+ acc
149
+ }
150
+
151
+ eval(&expr, vars, env_bpm, env_beat)
152
+ }
@@ -0,0 +1,16 @@
1
+ use crate::core::audio::evaluator::numeric::evaluate_numeric_expression;
2
+ use devalang_types::Value;
3
+ use devalang_types::VariableTable;
4
+
5
+ pub fn evaluate_rhs_into_value(
6
+ raw: &str,
7
+ vars: &VariableTable,
8
+ env_bpm: f32,
9
+ env_beat: f32,
10
+ ) -> Value {
11
+ if let Some(num) = evaluate_numeric_expression(raw, vars, env_bpm, env_beat) {
12
+ Value::Number(num)
13
+ } else {
14
+ Value::String(raw.to_string())
15
+ }
16
+ }
@@ -0,0 +1,94 @@
1
+ use crate::core::audio::evaluator::numeric::evaluate_numeric_expression;
2
+ use devalang_types::Value;
3
+ use devalang_types::VariableTable;
4
+
5
+ // Evaluate a simple string concatenation expression like: "hello " + name + "!" + $env.beat
6
+ // - Splits on + outside quotes
7
+ // - Terms can be string literals (double quotes), variables (Number/String/Boolean), or numeric env/math expressions
8
+ // Returns None if parsing fails (fallback to raw print)
9
+ pub fn evaluate_string_expression(
10
+ expr: &str,
11
+ vars: &VariableTable,
12
+ env_bpm: f32,
13
+ env_beat: f32,
14
+ ) -> Option<String> {
15
+ // Quick reject if no '+' present
16
+ if !expr.contains('+') {
17
+ return None;
18
+ }
19
+
20
+ // Split by '+' outside of quotes
21
+ let mut parts: Vec<String> = Vec::new();
22
+ let mut cur = String::new();
23
+ let mut in_quotes = false;
24
+ let mut escape = false;
25
+ for ch in expr.chars() {
26
+ if escape {
27
+ cur.push(ch);
28
+ escape = false;
29
+ continue;
30
+ }
31
+ if ch == '\\' {
32
+ // escape next char
33
+ escape = true;
34
+ continue;
35
+ }
36
+ if ch == '"' {
37
+ in_quotes = !in_quotes;
38
+ cur.push(ch);
39
+ continue;
40
+ }
41
+ if ch == '+' && !in_quotes {
42
+ parts.push(cur.to_string());
43
+ cur.clear();
44
+ continue;
45
+ }
46
+ cur.push(ch);
47
+ }
48
+ if !cur.is_empty() {
49
+ parts.push(cur.to_string());
50
+ }
51
+ if parts.is_empty() {
52
+ return None;
53
+ }
54
+
55
+ // Resolve each part into a string
56
+ fn strip_quotes(s: &str) -> Option<String> {
57
+ let st = s.trim();
58
+ if st.len() >= 2 && st.starts_with('"') && st.ends_with('"') {
59
+ Some(st[1..st.len() - 1].to_string())
60
+ } else {
61
+ None
62
+ }
63
+ }
64
+
65
+ let mut out = String::new();
66
+ for p in parts {
67
+ if p.is_empty() {
68
+ continue;
69
+ }
70
+ if let Some(lit) = strip_quotes(&p) {
71
+ out.push_str(&lit);
72
+ continue;
73
+ }
74
+ // Try variables first
75
+ if let Some(val) = vars.get(&p) {
76
+ match val {
77
+ Value::String(s) => out.push_str(s),
78
+ Value::Number(n) => out.push_str(&format!("{}", n)),
79
+ Value::Boolean(b) => out.push_str(&format!("{}", b)),
80
+ other => out.push_str(&format!("{:?}", other)),
81
+ }
82
+ continue;
83
+ }
84
+ // Try env/math/numeric expression for this term
85
+ if let Some(n) = evaluate_numeric_expression(&p, vars, env_bpm, env_beat) {
86
+ out.push_str(&format!("{}", n));
87
+ continue;
88
+ }
89
+ // Bareword not resolved: include as-is (safe fallback)
90
+ out.push_str(&p);
91
+ }
92
+
93
+ Some(out)
94
+ }