@devaloop/devalang 0.0.1-alpha.15 → 0.0.1-alpha.16-hotfix.0

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 (173) hide show
  1. package/.devalang +2 -0
  2. package/.github/workflows/ci.yml +92 -0
  3. package/Cargo.toml +60 -58
  4. package/README.md +1 -1
  5. package/docs/CHANGELOG.md +34 -1
  6. package/docs/CONTRIBUTING.md +101 -1
  7. package/docs/ROADMAP.md +1 -1
  8. package/docs/TODO.md +1 -1
  9. package/examples/automation.deva +1 -3
  10. package/examples/bank.deva +4 -4
  11. package/examples/events.deva +12 -0
  12. package/examples/function.deva +4 -4
  13. package/examples/index.deva +3 -5
  14. package/examples/loop.deva +5 -11
  15. package/examples/pattern.deva +8 -0
  16. package/examples/plugin.deva +12 -11
  17. package/examples/variables.deva +1 -1
  18. package/out-tsc/bin/index.js +51 -7
  19. package/out-tsc/index.js +3 -1
  20. package/out-tsc/scripts/postbuild.js +9 -10
  21. package/out-tsc/scripts/postinstall.js +49 -0
  22. package/package.json +12 -4
  23. package/project-version.json +3 -3
  24. package/rust/cli/bank.rs +462 -455
  25. package/rust/cli/build.rs +252 -199
  26. package/rust/cli/check.rs +221 -180
  27. package/rust/cli/driver.rs +297 -292
  28. package/rust/cli/generator.rs +1 -0
  29. package/rust/cli/init.rs +87 -79
  30. package/rust/cli/install.rs +35 -32
  31. package/rust/cli/login.rs +127 -134
  32. package/rust/cli/mod.rs +13 -11
  33. package/rust/cli/play.rs +1123 -218
  34. package/rust/cli/telemetry.rs +19 -0
  35. package/rust/cli/template.rs +69 -57
  36. package/rust/cli/update.rs +6 -4
  37. package/rust/common/api.rs +5 -5
  38. package/rust/common/mod.rs +3 -3
  39. package/rust/config/driver.rs +118 -94
  40. package/rust/config/loader.rs +165 -156
  41. package/rust/config/mod.rs +4 -2
  42. package/rust/config/settings.rs +91 -0
  43. package/rust/config/stats.rs +257 -0
  44. package/rust/core/audio/engine.rs +696 -659
  45. package/rust/core/audio/evaluator.rs +263 -132
  46. package/rust/core/audio/interpreter/arrow_call.rs +198 -187
  47. package/rust/core/audio/interpreter/call.rs +98 -95
  48. package/rust/core/audio/interpreter/condition.rs +70 -71
  49. package/rust/core/audio/interpreter/driver.rs +487 -231
  50. package/rust/core/audio/interpreter/function.rs +26 -21
  51. package/rust/core/audio/interpreter/let_.rs +38 -26
  52. package/rust/core/audio/interpreter/load.rs +18 -18
  53. package/rust/core/audio/interpreter/loop_.rs +113 -106
  54. package/rust/core/audio/interpreter/mod.rs +14 -14
  55. package/rust/core/audio/interpreter/sleep.rs +27 -28
  56. package/rust/core/audio/interpreter/spawn.rs +105 -102
  57. package/rust/core/audio/interpreter/tempo.rs +19 -16
  58. package/rust/core/audio/interpreter/trigger.rs +239 -210
  59. package/rust/core/audio/loader/mod.rs +1 -1
  60. package/rust/core/audio/loader/trigger.rs +100 -94
  61. package/rust/core/audio/mod.rs +7 -7
  62. package/rust/core/audio/player.rs +64 -64
  63. package/rust/core/audio/renderer.rs +56 -53
  64. package/rust/core/audio/special/easing.rs +189 -120
  65. package/rust/core/audio/special/env.rs +43 -41
  66. package/rust/core/audio/special/math.rs +102 -92
  67. package/rust/core/audio/special/mod.rs +9 -9
  68. package/rust/core/audio/special/modulator.rs +143 -120
  69. package/rust/core/builder/mod.rs +80 -85
  70. package/rust/core/debugger/lexer.rs +27 -27
  71. package/rust/core/debugger/mod.rs +24 -23
  72. package/rust/core/debugger/module.rs +55 -47
  73. package/rust/core/debugger/preprocessor.rs +27 -27
  74. package/rust/core/debugger/store.rs +40 -39
  75. package/rust/core/error/mod.rs +80 -69
  76. package/rust/core/lexer/handler/arrow.rs +82 -82
  77. package/rust/core/lexer/handler/at.rs +21 -21
  78. package/rust/core/lexer/handler/brace.rs +41 -41
  79. package/rust/core/lexer/handler/colon.rs +21 -21
  80. package/rust/core/lexer/handler/comment.rs +30 -30
  81. package/rust/core/lexer/handler/dot.rs +21 -21
  82. package/rust/core/lexer/handler/driver.rs +337 -292
  83. package/rust/core/lexer/handler/identifier.rs +46 -43
  84. package/rust/core/lexer/handler/indent.rs +66 -66
  85. package/rust/core/lexer/handler/mod.rs +16 -16
  86. package/rust/core/lexer/handler/newline.rs +23 -23
  87. package/rust/core/lexer/handler/number.rs +31 -31
  88. package/rust/core/lexer/handler/operator.rs +46 -46
  89. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  90. package/rust/core/lexer/handler/slash.rs +21 -21
  91. package/rust/core/lexer/handler/string.rs +63 -63
  92. package/rust/core/lexer/mod.rs +54 -51
  93. package/rust/core/lexer/token.rs +97 -94
  94. package/rust/core/mod.rs +11 -11
  95. package/rust/core/parser/driver.rs +513 -490
  96. package/rust/core/parser/handler/arrow_call.rs +233 -227
  97. package/rust/core/parser/handler/at.rs +245 -162
  98. package/rust/core/parser/handler/bank.rs +94 -69
  99. package/rust/core/parser/handler/condition.rs +80 -74
  100. package/rust/core/parser/handler/dot.rs +143 -135
  101. package/rust/core/parser/handler/identifier/automate.rs +257 -194
  102. package/rust/core/parser/handler/identifier/call.rs +91 -88
  103. package/rust/core/parser/handler/identifier/emit.rs +66 -0
  104. package/rust/core/parser/handler/identifier/function.rs +100 -91
  105. package/rust/core/parser/handler/identifier/group.rs +85 -75
  106. package/rust/core/parser/handler/identifier/let_.rs +158 -143
  107. package/rust/core/parser/handler/identifier/mod.rs +54 -56
  108. package/rust/core/parser/handler/identifier/on.rs +98 -0
  109. package/rust/core/parser/handler/identifier/print.rs +52 -29
  110. package/rust/core/parser/handler/identifier/sleep.rs +36 -33
  111. package/rust/core/parser/handler/identifier/spawn.rs +91 -88
  112. package/rust/core/parser/handler/identifier/synth.rs +65 -63
  113. package/rust/core/parser/handler/loop_.rs +170 -89
  114. package/rust/core/parser/handler/mod.rs +8 -8
  115. package/rust/core/parser/handler/tempo.rs +53 -47
  116. package/rust/core/parser/mod.rs +4 -4
  117. package/rust/core/parser/statement.rs +142 -113
  118. package/rust/core/plugin/loader.rs +123 -48
  119. package/rust/core/plugin/mod.rs +2 -1
  120. package/rust/core/plugin/runner.rs +296 -0
  121. package/rust/core/preprocessor/loader.rs +515 -326
  122. package/rust/core/preprocessor/mod.rs +4 -4
  123. package/rust/core/preprocessor/module.rs +60 -58
  124. package/rust/core/preprocessor/processor.rs +99 -101
  125. package/rust/core/preprocessor/resolver/bank.rs +51 -48
  126. package/rust/core/preprocessor/resolver/call.rs +100 -101
  127. package/rust/core/preprocessor/resolver/condition.rs +97 -97
  128. package/rust/core/preprocessor/resolver/driver.rs +310 -280
  129. package/rust/core/preprocessor/resolver/function.rs +69 -68
  130. package/rust/core/preprocessor/resolver/group.rs +96 -91
  131. package/rust/core/preprocessor/resolver/let_.rs +32 -28
  132. package/rust/core/preprocessor/resolver/loop_.rs +320 -121
  133. package/rust/core/preprocessor/resolver/mod.rs +15 -15
  134. package/rust/core/preprocessor/resolver/spawn.rs +76 -73
  135. package/rust/core/preprocessor/resolver/synth.rs +56 -50
  136. package/rust/core/preprocessor/resolver/tempo.rs +50 -49
  137. package/rust/core/preprocessor/resolver/trigger.rs +113 -115
  138. package/rust/core/preprocessor/resolver/value.rs +81 -81
  139. package/rust/core/shared/duration.rs +9 -9
  140. package/rust/core/shared/mod.rs +3 -3
  141. package/rust/core/shared/value.rs +35 -32
  142. package/rust/core/store/function.rs +34 -34
  143. package/rust/core/store/global.rs +55 -38
  144. package/rust/core/store/mod.rs +5 -5
  145. package/rust/core/store/variable.rs +37 -34
  146. package/rust/core/utils/mod.rs +2 -2
  147. package/rust/core/utils/path.rs +37 -31
  148. package/rust/core/utils/validation.rs +35 -36
  149. package/rust/installer/addon.rs +84 -80
  150. package/rust/installer/bank.rs +62 -65
  151. package/rust/installer/mod.rs +5 -5
  152. package/rust/installer/plugin.rs +54 -55
  153. package/rust/installer/utils.rs +56 -56
  154. package/rust/lib.rs +156 -164
  155. package/rust/main.rs +250 -144
  156. package/rust/utils/error.rs +200 -51
  157. package/rust/utils/file.rs +38 -35
  158. package/rust/utils/first_usage.rs +76 -0
  159. package/rust/utils/logger.rs +195 -143
  160. package/rust/utils/mod.rs +9 -7
  161. package/rust/utils/signature.rs +19 -17
  162. package/rust/utils/spinner.rs +22 -19
  163. package/rust/utils/telemetry.rs +292 -0
  164. package/rust/utils/watcher.rs +34 -33
  165. package/templates/minimal/README.md +97 -121
  166. package/templates/welcome/README.md +97 -121
  167. package/typescript/bin/index.ts +19 -5
  168. package/typescript/index.ts +3 -1
  169. package/typescript/scripts/postbuild.ts +10 -6
  170. package/typescript/scripts/postinstall.ts +56 -0
  171. package/typescript/scripts/version/bump.ts +0 -1
  172. package/typescript/scripts/version/index.ts +0 -1
  173. package/out-tsc/bin/devalang.exe +0 -0
@@ -1,132 +1,263 @@
1
- use crate::core::{ shared::value::Value, store::variable::VariableTable };
2
- use crate::core::audio::special::{
3
- resolve_env_atom,
4
- find_and_eval_first_math_call,
5
- find_and_eval_first_easing_call,
6
- find_and_eval_first_mod_call,
7
- };
8
-
9
- pub fn evaluate_condition_string(expr: &str, vars: &VariableTable) -> bool {
10
- let tokens: Vec<&str> = expr.split_whitespace().collect();
11
- if tokens.len() != 3 {
12
- return false;
13
- }
14
-
15
- let left = tokens[0];
16
- let op = tokens[1];
17
- let right = tokens[2];
18
-
19
- let left_val = match vars.get(left) {
20
- Some(Value::Number(n)) => *n,
21
- _ => {
22
- return false;
23
- }
24
- };
25
-
26
- let right_val: f32 = right.parse().unwrap_or(0.0);
27
-
28
- match op {
29
- ">" => left_val > right_val,
30
- "<" => left_val < right_val,
31
- ">=" => left_val >= right_val,
32
- "<=" => left_val <= right_val,
33
- "==" => (left_val - right_val).abs() < f32::EPSILON,
34
- "!=" => (left_val - right_val).abs() > f32::EPSILON,
35
- _ => false,
36
- }
37
- }
38
-
39
- // Very small expression evaluator for `$env.*`, `$math.*` and variables.
40
- // Supports: +, -, *, / and simple parentheses, left-to-right (no precedence), and $math.(sin|cos)(expr)
41
- pub fn evaluate_numeric_expression(expr: &str, vars: &VariableTable, env_bpm: f32, env_beat: f32) -> Option<f32> {
42
- let expr = expr.replace(" ", "");
43
-
44
- // Helper to resolve an atom to a number
45
- fn resolve_atom(atom: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
46
- if let Some(v) = resolve_env_atom(atom, bpm, beat) { return Some(v); }
47
- if let Ok(n) = atom.parse::<f32>() { return Some(n); }
48
- if let Some(Value::Number(n)) = vars.get(atom) { return Some(*n); }
49
- None
50
- }
51
-
52
- // Shunting-like, simplified: first evaluate any $math.func(...) calls anywhere in the expression,
53
- // then fold remaining parentheses and evaluate left-to-right.
54
- fn eval(expr: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
55
- // 1) Replace $math.* calls progressively
56
- let mut s = expr.to_string();
57
- // Evaluate modulators first (they may feed easing/math)
58
- while let Some(next) = find_and_eval_first_mod_call(&s, evaluate_numeric_expression, vars, bpm, beat) { s = next; }
59
- // Then easing functions
60
- while let Some(next) = find_and_eval_first_easing_call(&s, evaluate_numeric_expression, vars, bpm, beat) { s = next; }
61
- // Finally math transforms
62
- while let Some(next) = find_and_eval_first_math_call(&s, evaluate_numeric_expression, vars, bpm, beat) { s = next; }
63
-
64
- // 2) Evaluate remaining (pure) parentheses starting from innermost
65
- if let Some(open) = s.rfind('(') {
66
- if let Some(close_rel) = s[open..].find(')') { // index relatif
67
- let close = open + close_rel;
68
- let inner = &s[open + 1..close];
69
- let val = eval(inner, vars, bpm, beat)?;
70
- let mut replaced = String::new();
71
- replaced.push_str(&s[..open]);
72
- replaced.push_str(&val.to_string());
73
- replaced.push_str(&s[close + 1..]);
74
- return eval(&replaced, vars, bpm, beat);
75
- }
76
- }
77
-
78
- // Tokenize by operators left-to-right
79
- let mut parts: Vec<String> = Vec::new();
80
- let mut cur = String::new();
81
- for ch in s.chars() {
82
- if "+-*/".contains(ch) {
83
- if !cur.is_empty() { parts.push(cur.clone()); cur.clear(); }
84
- parts.push(ch.to_string());
85
- } else {
86
- cur.push(ch);
87
- }
88
- }
89
- if !cur.is_empty() { parts.push(cur); }
90
- if parts.is_empty() { return None; }
91
-
92
- // Resolve atoms and compute
93
- let mut acc: Option<f32> = None;
94
- let mut op: Option<char> = None;
95
- for part in parts {
96
- if part.len() == 1 && "+-*/".contains(part.chars().next().unwrap()) {
97
- op = part.chars().next();
98
- continue;
99
- }
100
- let val = if let Some(v) = resolve_atom(&part, vars, bpm, beat) {
101
- v
102
- } else if part.starts_with("$env.") {
103
- // $env atom not handled by resolve_atom (when composed), try recursive eval
104
- eval(&part, vars, bpm, beat)?
105
- } else {
106
- return None;
107
- };
108
-
109
- acc = Some(match (acc, op) {
110
- (None, _) => val,
111
- (Some(a), Some('+')) => a + val,
112
- (Some(a), Some('-')) => a - val,
113
- (Some(a), Some('*')) => a * val,
114
- (Some(a), Some('/')) => if val != 0.0 { a / val } else { return Some(f32::INFINITY); },
115
- (Some(_), None) => val,
116
- _ => return None,
117
- });
118
- }
119
-
120
- acc
121
- }
122
-
123
- eval(&expr, vars, env_bpm, env_beat)
124
- }
125
-
126
- pub fn evaluate_rhs_into_value(raw: &str, vars: &VariableTable, env_bpm: f32, env_beat: f32) -> Value {
127
- if let Some(num) = evaluate_numeric_expression(raw, vars, env_bpm, env_beat) {
128
- Value::Number(num)
129
- } else {
130
- Value::String(raw.to_string())
131
- }
132
- }
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::{shared::value::Value, store::variable::VariableTable};
6
+
7
+ pub fn evaluate_condition_string(expr: &str, vars: &VariableTable) -> bool {
8
+ let tokens: Vec<&str> = expr.split_whitespace().collect();
9
+ if tokens.len() != 3 {
10
+ return false;
11
+ }
12
+
13
+ let left = tokens[0];
14
+ let op = tokens[1];
15
+ let right = tokens[2];
16
+
17
+ let left_val = match vars.get(left) {
18
+ Some(Value::Number(n)) => *n,
19
+ _ => {
20
+ return false;
21
+ }
22
+ };
23
+
24
+ let right_val: f32 = right.parse().unwrap_or(0.0);
25
+
26
+ match op {
27
+ ">" => left_val > right_val,
28
+ "<" => left_val < right_val,
29
+ ">=" => left_val >= right_val,
30
+ "<=" => left_val <= right_val,
31
+ "==" => (left_val - right_val).abs() < f32::EPSILON,
32
+ "!=" => (left_val - right_val).abs() > f32::EPSILON,
33
+ _ => false,
34
+ }
35
+ }
36
+
37
+ // Very small expression evaluator for `$env.*`, `$math.*` and variables.
38
+ // Supports: +, -, *, / and simple parentheses, left-to-right (no precedence), and $math.(sin|cos)(expr)
39
+ pub fn evaluate_numeric_expression(
40
+ expr: &str,
41
+ vars: &VariableTable,
42
+ env_bpm: f32,
43
+ env_beat: f32,
44
+ ) -> Option<f32> {
45
+ let expr = expr.replace(" ", "");
46
+
47
+ // Helper to resolve an atom to a number
48
+ fn resolve_atom(atom: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
49
+ if let Some(v) = resolve_env_atom(atom, bpm, beat) {
50
+ return Some(v);
51
+ }
52
+ if let Ok(n) = atom.parse::<f32>() {
53
+ return Some(n);
54
+ }
55
+ if let Some(Value::Number(n)) = vars.get(atom) {
56
+ return Some(*n);
57
+ }
58
+ None
59
+ }
60
+
61
+ // Shunting-like, simplified: first evaluate any $math.func(...) calls anywhere in the expression,
62
+ // then fold remaining parentheses and evaluate left-to-right.
63
+ fn eval(expr: &str, vars: &VariableTable, bpm: f32, beat: f32) -> Option<f32> {
64
+ // 1) Replace $math.* calls progressively
65
+ let mut s = expr.to_string();
66
+ // Evaluate modulators first (they may feed easing/math)
67
+ while let Some(next) =
68
+ find_and_eval_first_mod_call(&s, evaluate_numeric_expression, vars, bpm, beat)
69
+ {
70
+ s = next;
71
+ }
72
+ // Then easing functions
73
+ while let Some(next) =
74
+ find_and_eval_first_easing_call(&s, evaluate_numeric_expression, vars, bpm, beat)
75
+ {
76
+ s = next;
77
+ }
78
+ // Finally math transforms
79
+ while let Some(next) =
80
+ find_and_eval_first_math_call(&s, evaluate_numeric_expression, vars, bpm, beat)
81
+ {
82
+ s = next;
83
+ }
84
+
85
+ // 2) Evaluate remaining (pure) parentheses starting from innermost
86
+ if let Some(open) = s.rfind('(') {
87
+ if let Some(close_rel) = s[open..].find(')') {
88
+ // index relatif
89
+ let close = open + close_rel;
90
+ let inner = &s[open + 1..close];
91
+ let val = eval(inner, vars, bpm, beat)?;
92
+ let mut replaced = String::new();
93
+ replaced.push_str(&s[..open]);
94
+ replaced.push_str(&val.to_string());
95
+ replaced.push_str(&s[close + 1..]);
96
+ return eval(&replaced, vars, bpm, beat);
97
+ }
98
+ }
99
+
100
+ // Tokenize by operators left-to-right
101
+ let mut parts: Vec<String> = Vec::new();
102
+ let mut cur = String::new();
103
+ for ch in s.chars() {
104
+ if "+-*/".contains(ch) {
105
+ if !cur.is_empty() {
106
+ parts.push(cur.clone());
107
+ cur.clear();
108
+ }
109
+ parts.push(ch.to_string());
110
+ } else {
111
+ cur.push(ch);
112
+ }
113
+ }
114
+ if !cur.is_empty() {
115
+ parts.push(cur);
116
+ }
117
+ if parts.is_empty() {
118
+ return None;
119
+ }
120
+
121
+ // Resolve atoms and compute
122
+ let mut acc: Option<f32> = None;
123
+ let mut op: Option<char> = None;
124
+ for part in parts {
125
+ if part.len() == 1 && "+-*/".contains(part.chars().next().unwrap()) {
126
+ op = part.chars().next();
127
+ continue;
128
+ }
129
+ let val = if let Some(v) = resolve_atom(&part, vars, bpm, beat) {
130
+ v
131
+ } else if part.starts_with("$env.") {
132
+ // $env atom not handled by resolve_atom (when composed), try recursive eval
133
+ eval(&part, vars, bpm, beat)?
134
+ } else {
135
+ return None;
136
+ };
137
+
138
+ acc = Some(match (acc, op) {
139
+ (None, _) => val,
140
+ (Some(a), Some('+')) => a + val,
141
+ (Some(a), Some('-')) => a - val,
142
+ (Some(a), Some('*')) => a * val,
143
+ (Some(a), Some('/')) => {
144
+ if val != 0.0 {
145
+ a / val
146
+ } else {
147
+ return Some(f32::INFINITY);
148
+ }
149
+ }
150
+ (Some(_), None) => val,
151
+ _ => return None,
152
+ });
153
+ }
154
+
155
+ acc
156
+ }
157
+
158
+ eval(&expr, vars, env_bpm, env_beat)
159
+ }
160
+
161
+ pub fn evaluate_rhs_into_value(
162
+ raw: &str,
163
+ vars: &VariableTable,
164
+ env_bpm: f32,
165
+ env_beat: f32,
166
+ ) -> Value {
167
+ if let Some(num) = evaluate_numeric_expression(raw, vars, env_bpm, env_beat) {
168
+ Value::Number(num)
169
+ } else {
170
+ Value::String(raw.to_string())
171
+ }
172
+ }
173
+
174
+ // Evaluate a simple string concatenation expression like: "hello " + name + "!" + $env.beat
175
+ // - Splits on + outside quotes
176
+ // - Terms can be string literals (double quotes), variables (Number/String/Boolean), or numeric env/math expressions
177
+ // Returns None if parsing fails (fallback to raw print)
178
+ pub fn evaluate_string_expression(
179
+ expr: &str,
180
+ vars: &VariableTable,
181
+ env_bpm: f32,
182
+ env_beat: f32,
183
+ ) -> Option<String> {
184
+ // Quick reject if no '+' present
185
+ if !expr.contains('+') {
186
+ return None;
187
+ }
188
+
189
+ // Split by '+' outside of quotes
190
+ let mut parts: Vec<String> = Vec::new();
191
+ let mut cur = String::new();
192
+ let mut in_quotes = false;
193
+ let mut escape = false;
194
+ for ch in expr.chars() {
195
+ if escape {
196
+ cur.push(ch);
197
+ escape = false;
198
+ continue;
199
+ }
200
+ if ch == '\\' {
201
+ // escape next char
202
+ escape = true;
203
+ continue;
204
+ }
205
+ if ch == '"' {
206
+ in_quotes = !in_quotes;
207
+ cur.push(ch);
208
+ continue;
209
+ }
210
+ if ch == '+' && !in_quotes {
211
+ parts.push(cur.to_string());
212
+ cur.clear();
213
+ continue;
214
+ }
215
+ cur.push(ch);
216
+ }
217
+ if !cur.is_empty() {
218
+ parts.push(cur.to_string());
219
+ }
220
+ if parts.is_empty() {
221
+ return None;
222
+ }
223
+
224
+ // Resolve each part into a string
225
+ fn strip_quotes(s: &str) -> Option<String> {
226
+ let st = s.trim();
227
+ if st.len() >= 2 && st.starts_with('"') && st.ends_with('"') {
228
+ Some(st[1..st.len() - 1].to_string())
229
+ } else {
230
+ None
231
+ }
232
+ }
233
+
234
+ let mut out = String::new();
235
+ for p in parts {
236
+ if p.is_empty() {
237
+ continue;
238
+ }
239
+ if let Some(lit) = strip_quotes(&p) {
240
+ out.push_str(&lit);
241
+ continue;
242
+ }
243
+ // Try variables first
244
+ if let Some(val) = vars.get(&p) {
245
+ match val {
246
+ crate::core::shared::value::Value::String(s) => out.push_str(s),
247
+ crate::core::shared::value::Value::Number(n) => out.push_str(&format!("{}", n)),
248
+ crate::core::shared::value::Value::Boolean(b) => out.push_str(&format!("{}", b)),
249
+ other => out.push_str(&format!("{:?}", other)),
250
+ }
251
+ continue;
252
+ }
253
+ // Try env/math/numeric expression for this term
254
+ if let Some(n) = evaluate_numeric_expression(&p, vars, env_bpm, env_beat) {
255
+ out.push_str(&format!("{}", n));
256
+ continue;
257
+ }
258
+ // Bareword not resolved: include as-is (safe fallback)
259
+ out.push_str(&p);
260
+ }
261
+
262
+ Some(out)
263
+ }