@devaloop/devalang 0.0.1-alpha.9 → 0.0.1-beta.2

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 (322) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +6 -1
  3. package/.github/workflows/ci.yml +103 -0
  4. package/Cargo.toml +81 -48
  5. package/README.md +137 -154
  6. package/docs/CHANGELOG.md +428 -1
  7. package/docs/CONTRIBUTING.md +101 -0
  8. package/docs/ROADMAP.md +14 -7
  9. package/docs/TODO.md +16 -15
  10. package/examples/automation.deva +42 -0
  11. package/examples/bank.deva +7 -0
  12. package/examples/bus.deva +10 -0
  13. package/examples/duration.deva +9 -0
  14. package/examples/effect.deva +2 -0
  15. package/examples/events.deva +12 -0
  16. package/examples/filter.deva +11 -0
  17. package/examples/function.deva +15 -0
  18. package/examples/index.deva +57 -12
  19. package/examples/lfo.deva +9 -0
  20. package/examples/loop.deva +5 -12
  21. package/examples/pattern.deva +8 -0
  22. package/examples/plugin.deva +16 -0
  23. package/examples/synth.deva +11 -1
  24. package/examples/synth_types.deva +17 -0
  25. package/examples/variables.deva +1 -1
  26. package/out-tsc/bin/index.d.ts +2 -0
  27. package/out-tsc/bin/index.js +51 -7
  28. package/out-tsc/core/functions/index.d.ts +42 -0
  29. package/out-tsc/core/functions/index.js +87 -0
  30. package/out-tsc/core/index.d.ts +6 -0
  31. package/out-tsc/core/index.js +22 -0
  32. package/out-tsc/core/types/index.d.ts +4 -0
  33. package/out-tsc/core/types/index.js +20 -0
  34. package/out-tsc/core/types/plugin.d.ts +18 -0
  35. package/out-tsc/core/types/plugin.js +2 -0
  36. package/out-tsc/core/types/result.d.ts +27 -0
  37. package/out-tsc/core/types/result.js +2 -0
  38. package/out-tsc/core/types/statement.d.ts +106 -0
  39. package/out-tsc/core/types/statement.js +2 -0
  40. package/out-tsc/core/types/value.d.ts +43 -0
  41. package/out-tsc/core/types/value.js +2 -0
  42. package/out-tsc/index.d.ts +7 -0
  43. package/out-tsc/index.js +42 -1
  44. package/out-tsc/pkg/devalang_core.d.ts +15 -0
  45. package/out-tsc/pkg/devalang_core.js +65 -0
  46. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +34 -0
  47. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  48. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  49. package/out-tsc/scripts/postinstall.d.ts +1 -0
  50. package/out-tsc/scripts/postinstall.js +83 -0
  51. package/out-tsc/scripts/version/bump.d.ts +1 -0
  52. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  53. package/out-tsc/scripts/version/index.d.ts +1 -0
  54. package/out-tsc/scripts/version/sync.d.ts +1 -0
  55. package/package.json +28 -7
  56. package/project-version.json +4 -4
  57. package/rust/cli/bank/api.rs +122 -0
  58. package/rust/cli/bank/commands.rs +306 -0
  59. package/rust/cli/bank/mod.rs +29 -0
  60. package/rust/cli/build/commands.rs +153 -0
  61. package/rust/cli/build/mod.rs +2 -0
  62. package/rust/cli/build/process.rs +165 -0
  63. package/rust/cli/check/mod.rs +208 -0
  64. package/rust/cli/discover/commands.rs +253 -0
  65. package/rust/cli/discover/config.rs +111 -0
  66. package/rust/cli/discover/fs.rs +19 -0
  67. package/rust/cli/discover/install.rs +103 -0
  68. package/rust/cli/discover/metadata.rs +48 -0
  69. package/rust/cli/discover/mod.rs +5 -0
  70. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  71. package/rust/cli/init/mod.rs +1 -0
  72. package/rust/cli/install/addon.rs +118 -0
  73. package/rust/cli/install/bank.rs +72 -0
  74. package/rust/cli/install/commands.rs +35 -0
  75. package/rust/cli/install/mod.rs +4 -0
  76. package/rust/cli/install/plugin.rs +80 -0
  77. package/rust/cli/login/commands.rs +124 -0
  78. package/rust/cli/login/mod.rs +1 -0
  79. package/rust/cli/mod.rs +9 -202
  80. package/rust/cli/parser.rs +359 -0
  81. package/rust/cli/play/commands.rs +375 -0
  82. package/rust/cli/play/io.rs +17 -0
  83. package/rust/cli/play/mod.rs +5 -0
  84. package/rust/cli/play/process.rs +159 -0
  85. package/rust/cli/play/realtime.rs +91 -0
  86. package/rust/cli/play/utils.rs +23 -0
  87. package/rust/cli/telemetry/commands.rs +22 -0
  88. package/rust/cli/telemetry/event_creator.rs +80 -0
  89. package/rust/cli/telemetry/mod.rs +3 -0
  90. package/rust/cli/telemetry/send.rs +51 -0
  91. package/rust/cli/{template.rs → template/commands.rs} +17 -5
  92. package/rust/cli/template/mod.rs +1 -0
  93. package/rust/cli/update/commands.rs +6 -0
  94. package/rust/cli/update/mod.rs +1 -0
  95. package/rust/config/driver.rs +112 -0
  96. package/rust/config/mod.rs +3 -16
  97. package/rust/config/ops.rs +26 -0
  98. package/rust/config/settings.rs +101 -0
  99. package/rust/core/audio/engine/driver.rs +220 -0
  100. package/rust/core/audio/engine/export.rs +169 -0
  101. package/rust/core/audio/engine/helpers.rs +178 -0
  102. package/rust/core/audio/engine/mod.rs +56 -0
  103. package/rust/core/audio/engine/notes/dsp.rs +85 -0
  104. package/rust/core/audio/engine/notes/mod.rs +44 -0
  105. package/rust/core/audio/engine/notes/params.rs +294 -0
  106. package/rust/core/audio/engine/sample/insert.rs +199 -0
  107. package/rust/core/audio/engine/sample/mod.rs +40 -0
  108. package/rust/core/audio/engine/sample/padding.rs +170 -0
  109. package/rust/core/audio/evaluator/condition.rs +61 -0
  110. package/rust/core/audio/evaluator/mod.rs +9 -0
  111. package/rust/core/audio/evaluator/numeric.rs +152 -0
  112. package/rust/core/audio/evaluator/rhs.rs +16 -0
  113. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  114. package/rust/core/audio/interpreter/driver.rs +574 -216
  115. package/rust/core/audio/interpreter/mod.rs +2 -12
  116. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
  117. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
  118. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
  119. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
  120. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  121. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  122. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  123. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  124. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  125. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  126. package/rust/core/audio/interpreter/statements/automate.rs +16 -0
  127. package/rust/core/audio/interpreter/statements/call.rs +295 -0
  128. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -69
  129. package/rust/core/audio/interpreter/statements/function.rs +24 -0
  130. package/rust/core/audio/interpreter/statements/let_.rs +36 -0
  131. package/rust/core/audio/interpreter/statements/load.rs +17 -0
  132. package/rust/core/audio/interpreter/statements/loop_.rs +115 -0
  133. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  134. package/rust/core/audio/interpreter/statements/sleep.rs +28 -0
  135. package/rust/core/audio/interpreter/statements/spawn.rs +253 -0
  136. package/rust/core/audio/interpreter/statements/tempo.rs +40 -0
  137. package/rust/core/audio/interpreter/statements/trigger.rs +239 -0
  138. package/rust/core/audio/loader/mod.rs +1 -1
  139. package/rust/core/audio/loader/trigger.rs +98 -52
  140. package/rust/core/audio/mod.rs +2 -2
  141. package/rust/core/audio/player.rs +28 -12
  142. package/rust/core/audio/special/easing.rs +189 -0
  143. package/rust/core/audio/special/env.rs +45 -0
  144. package/rust/core/audio/special/math.rs +134 -0
  145. package/rust/core/audio/special/mod.rs +9 -0
  146. package/rust/core/audio/special/modulator.rs +143 -0
  147. package/rust/core/builder/mod.rs +129 -80
  148. package/rust/core/debugger/lexer.rs +4 -4
  149. package/rust/core/debugger/logs.rs +52 -0
  150. package/rust/core/debugger/mod.rs +11 -2
  151. package/rust/core/debugger/preprocessor.rs +4 -4
  152. package/rust/core/debugger/store.rs +38 -25
  153. package/rust/core/error/mod.rs +221 -12
  154. package/rust/core/lexer/driver.rs +59 -0
  155. package/rust/core/lexer/handler/arrow.rs +62 -11
  156. package/rust/core/lexer/handler/at.rs +5 -5
  157. package/rust/core/lexer/handler/brace.rs +11 -11
  158. package/rust/core/lexer/handler/colon.rs +5 -5
  159. package/rust/core/lexer/handler/comment.rs +3 -3
  160. package/rust/core/lexer/handler/dot.rs +6 -6
  161. package/rust/core/lexer/handler/driver.rs +143 -32
  162. package/rust/core/lexer/handler/identifier.rs +11 -5
  163. package/rust/core/lexer/handler/indent.rs +18 -4
  164. package/rust/core/lexer/handler/mod.rs +6 -5
  165. package/rust/core/lexer/handler/newline.rs +3 -3
  166. package/rust/core/lexer/handler/number.rs +5 -5
  167. package/rust/core/lexer/handler/operator.rs +5 -3
  168. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  169. package/rust/core/lexer/handler/slash.rs +21 -0
  170. package/rust/core/lexer/handler/string.rs +3 -3
  171. package/rust/core/lexer/mod.rs +1 -49
  172. package/rust/core/lexer/token.rs +17 -12
  173. package/rust/core/mod.rs +9 -10
  174. package/rust/core/parser/driver/block.rs +111 -0
  175. package/rust/core/parser/driver/cursor.rs +82 -0
  176. package/rust/core/parser/driver/driver_impl.rs +139 -0
  177. package/rust/core/parser/driver/mod.rs +6 -0
  178. package/rust/core/parser/driver/parse_array.rs +120 -0
  179. package/rust/core/parser/driver/parse_map.rs +223 -0
  180. package/rust/core/parser/driver/parser.rs +160 -0
  181. package/rust/core/parser/handler/arrow_call.rs +277 -126
  182. package/rust/core/parser/handler/at.rs +142 -25
  183. package/rust/core/parser/handler/bank.rs +83 -20
  184. package/rust/core/parser/handler/condition.rs +14 -5
  185. package/rust/core/parser/handler/dot.rs +111 -75
  186. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  187. package/rust/core/parser/handler/identifier/call.rs +74 -24
  188. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  189. package/rust/core/parser/handler/identifier/function.rs +113 -0
  190. package/rust/core/parser/handler/identifier/group.rs +28 -14
  191. package/rust/core/parser/handler/identifier/let_.rs +61 -21
  192. package/rust/core/parser/handler/identifier/mod.rs +24 -20
  193. package/rust/core/parser/handler/identifier/on.rs +107 -0
  194. package/rust/core/parser/handler/identifier/print.rs +49 -0
  195. package/rust/core/parser/handler/identifier/sleep.rs +77 -14
  196. package/rust/core/parser/handler/identifier/spawn.rs +81 -31
  197. package/rust/core/parser/handler/identifier/synth.rs +102 -32
  198. package/rust/core/parser/handler/loop_.rs +144 -22
  199. package/rust/core/parser/handler/mod.rs +6 -5
  200. package/rust/core/parser/handler/pattern.rs +74 -0
  201. package/rust/core/parser/handler/tempo.rs +67 -9
  202. package/rust/core/parser/mod.rs +3 -4
  203. package/rust/core/parser/statement.rs +6 -92
  204. package/rust/core/plugin/loader.rs +137 -0
  205. package/rust/core/plugin/mod.rs +2 -0
  206. package/rust/core/plugin/runner/mod.rs +11 -0
  207. package/rust/core/plugin/runner/non_wasm.rs +297 -0
  208. package/rust/core/plugin/runner/wasm32.rs +43 -0
  209. package/rust/core/preprocessor/loader/inject.rs +278 -0
  210. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  211. package/rust/core/preprocessor/loader/mod.rs +235 -0
  212. package/rust/core/preprocessor/mod.rs +4 -4
  213. package/rust/core/preprocessor/module.rs +55 -50
  214. package/rust/core/preprocessor/processor/handlers.rs +107 -0
  215. package/rust/core/preprocessor/processor/mod.rs +1 -0
  216. package/rust/core/preprocessor/resolver/bank.rs +14 -12
  217. package/rust/core/preprocessor/resolver/call.rs +106 -105
  218. package/rust/core/preprocessor/resolver/condition.rs +13 -10
  219. package/rust/core/preprocessor/resolver/driver.rs +145 -48
  220. package/rust/core/preprocessor/resolver/function.rs +69 -0
  221. package/rust/core/preprocessor/resolver/group.rs +122 -61
  222. package/rust/core/preprocessor/resolver/let_.rs +13 -12
  223. package/rust/core/preprocessor/resolver/loop_.rs +240 -13
  224. package/rust/core/preprocessor/resolver/mod.rs +8 -6
  225. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  226. package/rust/core/preprocessor/resolver/spawn.rs +83 -42
  227. package/rust/core/preprocessor/resolver/synth.rs +15 -11
  228. package/rust/core/preprocessor/resolver/tempo.rs +13 -14
  229. package/rust/core/preprocessor/resolver/trigger.rs +32 -28
  230. package/rust/core/preprocessor/resolver/value.rs +111 -13
  231. package/rust/core/store/global.rs +57 -39
  232. package/rust/core/store/mod.rs +0 -3
  233. package/rust/lib.rs +323 -117
  234. package/rust/main.rs +388 -65
  235. package/rust/types/Cargo.toml +11 -0
  236. package/rust/types/src/addons.rs +55 -0
  237. package/rust/types/src/ast.rs +202 -0
  238. package/rust/types/src/config.rs +84 -0
  239. package/rust/types/src/lib.rs +15 -0
  240. package/rust/types/src/plugin.rs +20 -0
  241. package/rust/types/src/store.rs +139 -0
  242. package/rust/types/src/telemetry.rs +85 -0
  243. package/rust/utils/Cargo.toml +26 -0
  244. package/rust/utils/src/error.rs +186 -0
  245. package/rust/utils/src/file.rs +94 -0
  246. package/rust/utils/src/first_usage.rs +97 -0
  247. package/rust/utils/{mod.rs → src/lib.rs} +6 -3
  248. package/rust/utils/{logger.rs → src/logger.rs} +94 -17
  249. package/rust/utils/src/path.rs +129 -0
  250. package/rust/utils/src/signature.rs +41 -0
  251. package/rust/utils/{spinner.rs → src/spinner.rs} +7 -8
  252. package/rust/utils/src/version.rs +27 -0
  253. package/rust/utils/{watcher.rs → src/watcher.rs} +17 -4
  254. package/rust/web/api.rs +5 -0
  255. package/rust/web/cdn.rs +34 -0
  256. package/rust/web/mod.rs +3 -0
  257. package/rust/web/sso.rs +5 -0
  258. package/templates/minimal/README.md +143 -127
  259. package/templates/welcome/README.md +143 -127
  260. package/templates/welcome/src/index.deva +56 -8
  261. package/templates/welcome/src/variables.deva +2 -4
  262. package/tests/integration.rs +21 -0
  263. package/tests/rust/cli_check_build.rs +21 -0
  264. package/tests/rust/cli_help.rs +12 -0
  265. package/tests/rust/cli_template_list.rs +10 -0
  266. package/tests/rust/cli_version.rs +11 -0
  267. package/tests/typescript/index.spec.ts +136 -0
  268. package/tests/typescript/playhead.spec.ts +36 -0
  269. package/tests/typescript/render_e2e.spec.ts +77 -0
  270. package/tsconfig.json +12 -10
  271. package/typescript/bin/index.ts +19 -5
  272. package/typescript/core/functions/index.ts +94 -0
  273. package/typescript/core/index.ts +6 -0
  274. package/typescript/core/types/index.ts +4 -0
  275. package/typescript/core/types/plugin.ts +19 -0
  276. package/typescript/core/types/result.ts +29 -0
  277. package/typescript/core/types/statement.ts +47 -0
  278. package/typescript/core/types/value.ts +29 -0
  279. package/typescript/index.ts +8 -1
  280. package/typescript/pkg/devalang_core.d.ts +4 -0
  281. package/typescript/pkg/devalang_core.ts +65 -0
  282. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  283. package/typescript/scripts/postinstall.ts +85 -0
  284. package/typescript/scripts/version/bump.ts +0 -1
  285. package/typescript/scripts/version/index.ts +0 -1
  286. package/docs/COMMANDS.md +0 -85
  287. package/docs/CONFIG.md +0 -30
  288. package/docs/SYNTAX.md +0 -210
  289. package/out-tsc/bin/devalang.exe +0 -0
  290. package/out-tsc/scripts/postbuild.js +0 -11
  291. package/rust/cli/build.rs +0 -137
  292. package/rust/cli/check.rs +0 -117
  293. package/rust/cli/play.rs +0 -193
  294. package/rust/config/loader.rs +0 -13
  295. package/rust/core/audio/engine.rs +0 -203
  296. package/rust/core/audio/evaluator.rs +0 -31
  297. package/rust/core/audio/interpreter/arrow_call.rs +0 -129
  298. package/rust/core/audio/interpreter/call.rs +0 -64
  299. package/rust/core/audio/interpreter/let_.rs +0 -19
  300. package/rust/core/audio/interpreter/load.rs +0 -18
  301. package/rust/core/audio/interpreter/loop_.rs +0 -67
  302. package/rust/core/audio/interpreter/sleep.rs +0 -36
  303. package/rust/core/audio/interpreter/spawn.rs +0 -66
  304. package/rust/core/audio/interpreter/tempo.rs +0 -16
  305. package/rust/core/audio/interpreter/trigger.rs +0 -69
  306. package/rust/core/audio/renderer.rs +0 -54
  307. package/rust/core/parser/driver.rs +0 -331
  308. package/rust/core/preprocessor/loader.rs +0 -193
  309. package/rust/core/preprocessor/processor.rs +0 -76
  310. package/rust/core/shared/duration.rs +0 -8
  311. package/rust/core/shared/mod.rs +0 -2
  312. package/rust/core/shared/value.rs +0 -18
  313. package/rust/core/store/export.rs +0 -28
  314. package/rust/core/store/import.rs +0 -28
  315. package/rust/core/store/variable.rs +0 -28
  316. package/rust/core/utils/mod.rs +0 -2
  317. package/rust/core/utils/path.rs +0 -31
  318. package/rust/core/utils/validation.rs +0 -37
  319. package/rust/utils/file.rs +0 -35
  320. package/rust/utils/signature.rs +0 -17
  321. package/rust/utils/version.rs +0 -15
  322. package/typescript/scripts/postbuild.ts +0 -8
@@ -0,0 +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
+ }
@@ -0,0 +1,9 @@
1
+ pub mod condition;
2
+ pub mod numeric;
3
+ pub mod rhs;
4
+ pub mod string_expr;
5
+
6
+ pub use condition::evaluate_condition_string;
7
+ pub use numeric::evaluate_numeric_expression;
8
+ pub use rhs::evaluate_rhs_into_value;
9
+ pub use string_expr::evaluate_string_expression;
@@ -0,0 +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
+ }
@@ -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
+ }