@devaloop/devalang 0.0.1-beta.2 → 0.0.1-beta.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (159) hide show
  1. package/Cargo.toml +84 -81
  2. package/README.md +3 -2
  3. package/docs/CHANGELOG.md +41 -0
  4. package/docs/ROADMAP.md +3 -3
  5. package/examples/chain.deva +19 -0
  6. package/examples/plugin.deva +10 -10
  7. package/examples/routing.deva +23 -0
  8. package/out-tsc/bin/project-version.json +6 -0
  9. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -8
  10. package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
  11. package/out-tsc/scripts/version/copy-to-binary.js +79 -0
  12. package/package.json +23 -10
  13. package/project-version.json +3 -3
  14. package/rust/bindings/Cargo.toml +9 -0
  15. package/rust/bindings/src/lib.rs +86 -0
  16. package/rust/cli/addon/commands.rs +35 -0
  17. package/rust/cli/addon/download.rs +234 -0
  18. package/rust/cli/addon/install.rs +33 -0
  19. package/rust/cli/addon/list.rs +224 -0
  20. package/rust/cli/addon/metadata.rs +124 -0
  21. package/rust/cli/addon/mod.rs +8 -0
  22. package/rust/cli/addon/remove.rs +271 -0
  23. package/rust/cli/addon/update.rs +305 -0
  24. package/rust/cli/{install/addon.rs → addon/utils.rs} +109 -118
  25. package/rust/cli/build/commands.rs +153 -153
  26. package/rust/cli/build/process.rs +165 -165
  27. package/rust/cli/check/mod.rs +208 -208
  28. package/rust/cli/discover/commands.rs +275 -253
  29. package/rust/cli/discover/config.rs +109 -111
  30. package/rust/cli/discover/fs.rs +19 -19
  31. package/rust/cli/discover/install.rs +214 -103
  32. package/rust/cli/discover/metadata.rs +48 -48
  33. package/rust/cli/discover/mod.rs +5 -5
  34. package/rust/cli/me/commands.rs +52 -0
  35. package/rust/cli/me/mod.rs +1 -0
  36. package/rust/cli/mod.rs +12 -12
  37. package/rust/cli/parser.rs +30 -69
  38. package/rust/cli/play/commands.rs +375 -375
  39. package/rust/cli/play/process.rs +159 -159
  40. package/rust/core/audio/engine/driver.rs +19 -2
  41. package/rust/core/audio/engine/export.rs +169 -169
  42. package/rust/core/audio/engine/mod.rs +56 -56
  43. package/rust/core/audio/engine/notes/dsp.rs +88 -85
  44. package/rust/core/audio/engine/notes/mod.rs +53 -44
  45. package/rust/core/audio/engine/notes/params.rs +294 -294
  46. package/rust/core/audio/engine/sample/insert.rs +148 -47
  47. package/rust/core/audio/engine/sample/mod.rs +40 -40
  48. package/rust/core/audio/engine/sample/padding.rs +170 -170
  49. package/rust/core/audio/evaluator/condition.rs +61 -61
  50. package/rust/core/audio/evaluator/numeric.rs +152 -152
  51. package/rust/core/audio/evaluator/rhs.rs +16 -16
  52. package/rust/core/audio/evaluator/string_expr.rs +94 -94
  53. package/rust/core/audio/interpreter/driver.rs +574 -574
  54. package/rust/core/audio/interpreter/mod.rs +2 -2
  55. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +9 -5
  56. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -384
  57. package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
  58. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +1 -0
  59. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +66 -11
  60. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -3
  61. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -192
  62. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -24
  63. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -116
  64. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -97
  65. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -100
  66. package/rust/core/audio/interpreter/statements/automate.rs +16 -16
  67. package/rust/core/audio/interpreter/statements/call.rs +31 -1
  68. package/rust/core/audio/interpreter/statements/condition.rs +72 -72
  69. package/rust/core/audio/interpreter/statements/function.rs +24 -24
  70. package/rust/core/audio/interpreter/statements/let_.rs +36 -36
  71. package/rust/core/audio/interpreter/statements/load.rs +17 -17
  72. package/rust/core/audio/interpreter/statements/loop_.rs +115 -115
  73. package/rust/core/audio/interpreter/statements/spawn.rs +51 -2
  74. package/rust/core/audio/interpreter/statements/trigger.rs +242 -239
  75. package/rust/core/audio/loader/trigger.rs +98 -98
  76. package/rust/core/audio/player.rs +70 -70
  77. package/rust/core/audio/special/mod.rs +9 -9
  78. package/rust/core/builder/mod.rs +129 -129
  79. package/rust/core/debugger/lexer.rs +27 -27
  80. package/rust/core/debugger/logs.rs +52 -52
  81. package/rust/core/debugger/preprocessor.rs +27 -27
  82. package/rust/core/debugger/store.rs +38 -38
  83. package/rust/core/lexer/driver.rs +59 -59
  84. package/rust/core/lexer/handler/arrow.rs +82 -82
  85. package/rust/core/lexer/handler/at.rs +21 -21
  86. package/rust/core/lexer/handler/brace.rs +41 -41
  87. package/rust/core/lexer/handler/colon.rs +21 -21
  88. package/rust/core/lexer/handler/comment.rs +30 -30
  89. package/rust/core/lexer/handler/dot.rs +21 -21
  90. package/rust/core/lexer/handler/driver.rs +337 -337
  91. package/rust/core/lexer/handler/identifier.rs +47 -47
  92. package/rust/core/lexer/handler/indent.rs +66 -66
  93. package/rust/core/lexer/handler/mod.rs +15 -15
  94. package/rust/core/lexer/handler/newline.rs +23 -23
  95. package/rust/core/lexer/handler/number.rs +31 -31
  96. package/rust/core/lexer/handler/operator.rs +46 -46
  97. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  98. package/rust/core/lexer/handler/slash.rs +21 -21
  99. package/rust/core/lexer/handler/string.rs +63 -63
  100. package/rust/core/lexer/mod.rs +3 -3
  101. package/rust/core/mod.rs +9 -9
  102. package/rust/core/parser/driver/block.rs +111 -111
  103. package/rust/core/parser/driver/cursor.rs +82 -82
  104. package/rust/core/parser/driver/driver_impl.rs +21 -1
  105. package/rust/core/parser/driver/mod.rs +6 -6
  106. package/rust/core/parser/driver/parse_array.rs +120 -120
  107. package/rust/core/parser/driver/parse_map.rs +247 -223
  108. package/rust/core/parser/driver/parser.rs +160 -160
  109. package/rust/core/parser/handler/arrow_call.rs +65 -14
  110. package/rust/core/parser/handler/identifier/synth.rs +171 -135
  111. package/rust/core/parser/handler/mod.rs +9 -9
  112. package/rust/core/parser/handler/pattern.rs +24 -1
  113. package/rust/core/plugin/loader.rs +137 -137
  114. package/rust/core/plugin/mod.rs +2 -2
  115. package/rust/core/plugin/runner/non_wasm.rs +481 -297
  116. package/rust/core/plugin/runner/wasm32.rs +1 -0
  117. package/rust/core/preprocessor/loader/inject.rs +313 -278
  118. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -110
  119. package/rust/core/preprocessor/loader/mod.rs +235 -235
  120. package/rust/core/preprocessor/module.rs +55 -55
  121. package/rust/core/preprocessor/processor/handlers.rs +107 -107
  122. package/rust/core/preprocessor/resolver/bank.rs +49 -49
  123. package/rust/core/preprocessor/resolver/call.rs +124 -124
  124. package/rust/core/preprocessor/resolver/condition.rs +95 -95
  125. package/rust/core/preprocessor/resolver/driver.rs +324 -324
  126. package/rust/core/preprocessor/resolver/function.rs +69 -69
  127. package/rust/core/preprocessor/resolver/group.rs +122 -122
  128. package/rust/core/preprocessor/resolver/let_.rs +32 -32
  129. package/rust/core/preprocessor/resolver/loop_.rs +318 -318
  130. package/rust/core/preprocessor/resolver/mod.rs +16 -16
  131. package/rust/core/preprocessor/resolver/pattern.rs +95 -83
  132. package/rust/core/preprocessor/resolver/spawn.rs +99 -99
  133. package/rust/core/preprocessor/resolver/synth.rs +54 -54
  134. package/rust/core/preprocessor/resolver/tempo.rs +48 -48
  135. package/rust/core/preprocessor/resolver/trigger.rs +116 -116
  136. package/rust/core/preprocessor/resolver/value.rs +176 -176
  137. package/rust/core/store/global.rs +57 -57
  138. package/rust/lib.rs +323 -323
  139. package/rust/macros/Cargo.toml +14 -0
  140. package/rust/macros/src/lib.rs +52 -0
  141. package/rust/main.rs +311 -142
  142. package/rust/types/Cargo.toml +1 -1
  143. package/rust/types/src/addons.rs +3 -1
  144. package/rust/types/src/config.rs +1 -3
  145. package/rust/utils/Cargo.toml +5 -2
  146. package/rust/utils/src/file.rs +397 -14
  147. package/rust/utils/src/path.rs +31 -2
  148. package/rust/utils/src/version.rs +38 -7
  149. package/rust/web/auth.rs +5 -0
  150. package/rust/web/forge.rs +5 -0
  151. package/rust/web/mod.rs +5 -3
  152. package/typescript/scripts/version/copy-to-binary.ts +82 -0
  153. package/rust/cli/bank/api.rs +0 -122
  154. package/rust/cli/bank/commands.rs +0 -306
  155. package/rust/cli/bank/mod.rs +0 -29
  156. package/rust/cli/install/bank.rs +0 -72
  157. package/rust/cli/install/commands.rs +0 -35
  158. package/rust/cli/install/mod.rs +0 -4
  159. package/rust/cli/install/plugin.rs +0 -80
@@ -1,61 +1,61 @@
1
- use crate::core::audio::special::resolve_env_atom;
2
- use devalang_types::Value;
3
- use devalang_types::VariableTable;
4
-
5
- pub fn evaluate_condition_string(expr: &str, vars: &VariableTable) -> bool {
6
- let tokens: Vec<&str> = expr.split_whitespace().collect();
7
- if tokens.len() != 3 {
8
- return false;
9
- }
10
-
11
- let left = tokens[0];
12
- let op = tokens[1];
13
- let right = tokens[2];
14
-
15
- // Resolve left and right to numeric values where possible. Accept numbers, variables or env atoms.
16
- fn resolve_for_cond(s: &str, vars: &VariableTable) -> Option<f32> {
17
- if let Ok(n) = s.parse::<f32>() {
18
- return Some(n);
19
- }
20
- if let Some(Value::Number(n)) = vars.get(s) {
21
- return Some(*n);
22
- }
23
- if let Some(v) = resolve_env_atom(s, 120.0, 1.0) {
24
- return Some(v);
25
- }
26
- None
27
- }
28
-
29
- let left_val = match resolve_for_cond(left, vars) {
30
- Some(v) => v,
31
- None => {
32
- return false;
33
- }
34
- };
35
-
36
- let right_val = match resolve_for_cond(right, vars) {
37
- Some(v) => v,
38
- None => {
39
- return false;
40
- }
41
- };
42
-
43
- match op {
44
- ">" => left_val > right_val,
45
- "<" => left_val < right_val,
46
- ">=" => left_val >= right_val,
47
- "<=" => left_val <= right_val,
48
- "==" => {
49
- // relative epsilon for floating comparisons
50
- let diff = (left_val - right_val).abs();
51
- let largest = left_val.abs().max(right_val.abs()).max(1.0);
52
- diff <= f32::EPSILON * largest
53
- }
54
- "!=" => {
55
- let diff = (left_val - right_val).abs();
56
- let largest = left_val.abs().max(right_val.abs()).max(1.0);
57
- diff > f32::EPSILON * largest
58
- }
59
- _ => false,
60
- }
61
- }
1
+ use crate::core::audio::special::resolve_env_atom;
2
+ use devalang_types::Value;
3
+ use devalang_types::VariableTable;
4
+
5
+ pub fn evaluate_condition_string(expr: &str, vars: &VariableTable) -> bool {
6
+ let tokens: Vec<&str> = expr.split_whitespace().collect();
7
+ if tokens.len() != 3 {
8
+ return false;
9
+ }
10
+
11
+ let left = tokens[0];
12
+ let op = tokens[1];
13
+ let right = tokens[2];
14
+
15
+ // Resolve left and right to numeric values where possible. Accept numbers, variables or env atoms.
16
+ fn resolve_for_cond(s: &str, vars: &VariableTable) -> Option<f32> {
17
+ if let Ok(n) = s.parse::<f32>() {
18
+ return Some(n);
19
+ }
20
+ if let Some(Value::Number(n)) = vars.get(s) {
21
+ return Some(*n);
22
+ }
23
+ if let Some(v) = resolve_env_atom(s, 120.0, 1.0) {
24
+ return Some(v);
25
+ }
26
+ None
27
+ }
28
+
29
+ let left_val = match resolve_for_cond(left, vars) {
30
+ Some(v) => v,
31
+ None => {
32
+ return false;
33
+ }
34
+ };
35
+
36
+ let right_val = match resolve_for_cond(right, vars) {
37
+ Some(v) => v,
38
+ None => {
39
+ return false;
40
+ }
41
+ };
42
+
43
+ match op {
44
+ ">" => left_val > right_val,
45
+ "<" => left_val < right_val,
46
+ ">=" => left_val >= right_val,
47
+ "<=" => left_val <= right_val,
48
+ "==" => {
49
+ // relative epsilon for floating comparisons
50
+ let diff = (left_val - right_val).abs();
51
+ let largest = left_val.abs().max(right_val.abs()).max(1.0);
52
+ diff <= f32::EPSILON * largest
53
+ }
54
+ "!=" => {
55
+ let diff = (left_val - right_val).abs();
56
+ let largest = left_val.abs().max(right_val.abs()).max(1.0);
57
+ diff > f32::EPSILON * largest
58
+ }
59
+ _ => false,
60
+ }
61
+ }
@@ -1,152 +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 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
- }
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
+ }
@@ -1,16 +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
- }
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
+ }
@@ -1,94 +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
- }
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
+ }