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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (277) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +10 -4
  3. package/.github/workflows/ci.yml +103 -0
  4. package/Cargo.toml +80 -48
  5. package/README.md +135 -158
  6. package/docs/CHANGELOG.md +413 -1
  7. package/docs/CONTRIBUTING.md +101 -0
  8. package/docs/ROADMAP.md +10 -7
  9. package/docs/TODO.md +21 -9
  10. package/examples/automation.deva +42 -0
  11. package/examples/bank.deva +7 -0
  12. package/examples/condition.deva +8 -12
  13. package/examples/duration.deva +9 -0
  14. package/examples/events.deva +12 -0
  15. package/examples/function.deva +15 -0
  16. package/examples/group.deva +3 -3
  17. package/examples/index.deva +57 -10
  18. package/examples/loop.deva +7 -12
  19. package/examples/pattern.deva +8 -0
  20. package/examples/plugin.deva +16 -0
  21. package/examples/synth.deva +14 -0
  22. package/examples/variables.deva +2 -2
  23. package/out-tsc/bin/index.d.ts +2 -0
  24. package/out-tsc/bin/index.js +51 -7
  25. package/out-tsc/core/functions/index.d.ts +37 -0
  26. package/out-tsc/core/functions/index.js +76 -0
  27. package/out-tsc/core/index.d.ts +6 -0
  28. package/out-tsc/core/index.js +22 -0
  29. package/out-tsc/core/types/index.d.ts +4 -0
  30. package/out-tsc/core/types/index.js +20 -0
  31. package/out-tsc/core/types/plugin.d.ts +18 -0
  32. package/out-tsc/core/types/plugin.js +2 -0
  33. package/out-tsc/core/types/result.d.ts +27 -0
  34. package/out-tsc/core/types/result.js +2 -0
  35. package/out-tsc/core/types/statement.d.ts +106 -0
  36. package/out-tsc/core/types/statement.js +2 -0
  37. package/out-tsc/core/types/value.d.ts +43 -0
  38. package/out-tsc/core/types/value.js +2 -0
  39. package/out-tsc/index.d.ts +7 -0
  40. package/out-tsc/index.js +42 -1
  41. package/out-tsc/pkg/devalang_core.d.ts +13 -0
  42. package/out-tsc/pkg/devalang_core.js +50 -0
  43. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +33 -0
  44. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  45. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  46. package/out-tsc/scripts/postinstall.d.ts +1 -0
  47. package/out-tsc/scripts/postinstall.js +83 -0
  48. package/out-tsc/scripts/version/bump.d.ts +1 -0
  49. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  50. package/out-tsc/scripts/version/fetch.js +1 -5
  51. package/out-tsc/scripts/version/index.d.ts +1 -0
  52. package/out-tsc/scripts/version/sync.d.ts +1 -0
  53. package/package.json +28 -7
  54. package/project-version.json +4 -4
  55. package/rust/cli/bank/api.rs +122 -0
  56. package/rust/cli/bank/commands.rs +275 -0
  57. package/rust/cli/bank/mod.rs +29 -0
  58. package/rust/cli/build/commands.rs +103 -0
  59. package/rust/cli/build/mod.rs +2 -0
  60. package/rust/cli/build/process.rs +146 -0
  61. package/rust/cli/check/mod.rs +208 -0
  62. package/rust/cli/discover/commands.rs +253 -0
  63. package/rust/cli/discover/config.rs +111 -0
  64. package/rust/cli/discover/fs.rs +19 -0
  65. package/rust/cli/discover/install.rs +103 -0
  66. package/rust/cli/discover/metadata.rs +48 -0
  67. package/rust/cli/discover/mod.rs +5 -0
  68. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  69. package/rust/cli/init/mod.rs +1 -0
  70. package/rust/cli/install/addon.rs +118 -0
  71. package/rust/cli/install/bank.rs +53 -0
  72. package/rust/cli/install/commands.rs +35 -0
  73. package/rust/cli/install/mod.rs +4 -0
  74. package/rust/cli/install/plugin.rs +61 -0
  75. package/rust/cli/login/commands.rs +124 -0
  76. package/rust/cli/login/mod.rs +1 -0
  77. package/rust/cli/mod.rs +12 -205
  78. package/rust/cli/parser.rs +314 -0
  79. package/rust/cli/play/commands.rs +324 -0
  80. package/rust/cli/play/io.rs +17 -0
  81. package/rust/cli/play/mod.rs +5 -0
  82. package/rust/cli/play/process.rs +150 -0
  83. package/rust/cli/play/realtime.rs +91 -0
  84. package/rust/cli/play/utils.rs +23 -0
  85. package/rust/cli/telemetry/commands.rs +22 -0
  86. package/rust/cli/telemetry/event_creator.rs +80 -0
  87. package/rust/cli/telemetry/mod.rs +3 -0
  88. package/rust/cli/telemetry/send.rs +51 -0
  89. package/rust/cli/{template.rs → template/commands.rs} +69 -57
  90. package/rust/cli/template/mod.rs +1 -0
  91. package/rust/cli/update/commands.rs +6 -0
  92. package/rust/cli/update/mod.rs +1 -0
  93. package/rust/config/driver.rs +103 -0
  94. package/rust/config/mod.rs +3 -16
  95. package/rust/config/ops.rs +26 -0
  96. package/rust/config/settings.rs +101 -0
  97. package/rust/core/audio/engine/helpers.rs +170 -0
  98. package/rust/core/audio/engine/mod.rs +7 -0
  99. package/rust/core/audio/engine/sample.rs +366 -0
  100. package/rust/core/audio/engine/synth.rs +325 -0
  101. package/rust/core/audio/evaluator.rs +310 -31
  102. package/rust/core/audio/interpreter/arrow_call.rs +311 -0
  103. package/rust/core/audio/interpreter/automate.rs +18 -0
  104. package/rust/core/audio/interpreter/call.rs +294 -42
  105. package/rust/core/audio/interpreter/condition.rs +71 -65
  106. package/rust/core/audio/interpreter/driver.rs +542 -204
  107. package/rust/core/audio/interpreter/function.rs +26 -0
  108. package/rust/core/audio/interpreter/let_.rs +38 -19
  109. package/rust/core/audio/interpreter/load.rs +19 -18
  110. package/rust/core/audio/interpreter/loop_.rs +114 -59
  111. package/rust/core/audio/interpreter/mod.rs +14 -11
  112. package/rust/core/audio/interpreter/sleep.rs +28 -36
  113. package/rust/core/audio/interpreter/spawn.rs +252 -65
  114. package/rust/core/audio/interpreter/tempo.rs +40 -16
  115. package/rust/core/audio/interpreter/trigger.rs +239 -69
  116. package/rust/core/audio/loader/mod.rs +1 -1
  117. package/rust/core/audio/loader/trigger.rs +97 -52
  118. package/rust/core/audio/mod.rs +7 -6
  119. package/rust/core/audio/player.rs +70 -54
  120. package/rust/core/audio/renderer.rs +54 -57
  121. package/rust/core/audio/special/easing.rs +189 -0
  122. package/rust/core/audio/special/env.rs +45 -0
  123. package/rust/core/audio/special/math.rs +134 -0
  124. package/rust/core/audio/special/mod.rs +9 -0
  125. package/rust/core/audio/special/modulator.rs +143 -0
  126. package/rust/core/builder/mod.rs +86 -80
  127. package/rust/core/debugger/lexer.rs +27 -27
  128. package/rust/core/debugger/mod.rs +30 -21
  129. package/rust/core/debugger/module.rs +55 -0
  130. package/rust/core/debugger/preprocessor.rs +27 -27
  131. package/rust/core/debugger/store.rs +40 -25
  132. package/rust/core/error/mod.rs +269 -60
  133. package/rust/core/lexer/driver.rs +61 -0
  134. package/rust/core/lexer/handler/arrow.rs +82 -0
  135. package/rust/core/lexer/handler/at.rs +21 -21
  136. package/rust/core/lexer/handler/brace.rs +41 -41
  137. package/rust/core/lexer/handler/colon.rs +21 -21
  138. package/rust/core/lexer/handler/comment.rs +30 -30
  139. package/rust/core/lexer/handler/dot.rs +21 -21
  140. package/rust/core/lexer/handler/driver.rs +337 -215
  141. package/rust/core/lexer/handler/identifier.rs +47 -40
  142. package/rust/core/lexer/handler/indent.rs +66 -52
  143. package/rust/core/lexer/handler/mod.rs +15 -13
  144. package/rust/core/lexer/handler/newline.rs +23 -23
  145. package/rust/core/lexer/handler/number.rs +31 -31
  146. package/rust/core/lexer/handler/operator.rs +46 -44
  147. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  148. package/rust/core/lexer/handler/slash.rs +21 -0
  149. package/rust/core/lexer/handler/string.rs +63 -63
  150. package/rust/core/lexer/mod.rs +3 -30
  151. package/rust/core/lexer/token.rs +21 -12
  152. package/rust/core/mod.rs +10 -10
  153. package/rust/core/parser/driver.rs +584 -312
  154. package/rust/core/parser/handler/arrow_call.rs +253 -0
  155. package/rust/core/parser/handler/at.rs +279 -162
  156. package/rust/core/parser/handler/bank.rs +104 -41
  157. package/rust/core/parser/handler/condition.rs +83 -74
  158. package/rust/core/parser/handler/dot.rs +148 -112
  159. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  160. package/rust/core/parser/handler/identifier/call.rs +91 -0
  161. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  162. package/rust/core/parser/handler/identifier/function.rs +113 -0
  163. package/rust/core/parser/handler/identifier/group.rs +89 -0
  164. package/rust/core/parser/handler/identifier/let_.rs +173 -0
  165. package/rust/core/parser/handler/identifier/mod.rs +55 -0
  166. package/rust/core/parser/handler/identifier/on.rs +107 -0
  167. package/rust/core/parser/handler/identifier/print.rs +49 -0
  168. package/rust/core/parser/handler/identifier/sleep.rs +43 -0
  169. package/rust/core/parser/handler/identifier/spawn.rs +91 -0
  170. package/rust/core/parser/handler/identifier/synth.rs +135 -0
  171. package/rust/core/parser/handler/loop_.rs +194 -66
  172. package/rust/core/parser/handler/mod.rs +9 -7
  173. package/rust/core/parser/handler/pattern.rs +74 -0
  174. package/rust/core/parser/handler/tempo.rs +57 -47
  175. package/rust/core/parser/mod.rs +3 -4
  176. package/rust/core/parser/statement.rs +11 -88
  177. package/rust/core/plugin/loader.rs +137 -0
  178. package/rust/core/plugin/mod.rs +2 -0
  179. package/rust/core/plugin/runner.rs +347 -0
  180. package/rust/core/preprocessor/loader.rs +637 -179
  181. package/rust/core/preprocessor/mod.rs +4 -4
  182. package/rust/core/preprocessor/module.rs +60 -53
  183. package/rust/core/preprocessor/processor.rs +114 -67
  184. package/rust/core/preprocessor/resolver/bank.rs +49 -47
  185. package/rust/core/preprocessor/resolver/call.rs +124 -53
  186. package/rust/core/preprocessor/resolver/condition.rs +95 -66
  187. package/rust/core/preprocessor/resolver/driver.rs +324 -182
  188. package/rust/core/preprocessor/resolver/function.rs +69 -0
  189. package/rust/core/preprocessor/resolver/group.rs +94 -118
  190. package/rust/core/preprocessor/resolver/let_.rs +32 -0
  191. package/rust/core/preprocessor/resolver/loop_.rs +318 -145
  192. package/rust/core/preprocessor/resolver/mod.rs +16 -10
  193. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  194. package/rust/core/preprocessor/resolver/spawn.rs +99 -53
  195. package/rust/core/preprocessor/resolver/synth.rs +54 -0
  196. package/rust/core/preprocessor/resolver/tempo.rs +48 -49
  197. package/rust/core/preprocessor/resolver/trigger.rs +116 -111
  198. package/rust/core/preprocessor/resolver/value.rs +176 -0
  199. package/rust/core/store/export.rs +28 -28
  200. package/rust/core/store/function.rs +40 -0
  201. package/rust/core/store/global.rs +61 -39
  202. package/rust/core/store/import.rs +28 -28
  203. package/rust/core/store/mod.rs +5 -4
  204. package/rust/core/store/variable.rs +51 -28
  205. package/rust/core/utils/mod.rs +1 -2
  206. package/rust/core/utils/path.rs +37 -46
  207. package/rust/lib.rs +308 -117
  208. package/rust/main.rs +364 -65
  209. package/rust/types/Cargo.toml +11 -0
  210. package/rust/types/src/addons.rs +55 -0
  211. package/rust/types/src/ast.rs +202 -0
  212. package/rust/types/src/config.rs +74 -0
  213. package/rust/types/src/lib.rs +12 -0
  214. package/rust/types/src/telemetry.rs +85 -0
  215. package/rust/utils/Cargo.toml +26 -0
  216. package/rust/utils/src/error.rs +186 -0
  217. package/rust/utils/src/file.rs +94 -0
  218. package/rust/utils/src/first_usage.rs +97 -0
  219. package/rust/utils/{mod.rs → src/lib.rs} +9 -6
  220. package/rust/utils/{logger.rs → src/logger.rs} +200 -123
  221. package/rust/utils/src/path.rs +88 -0
  222. package/rust/utils/src/signature.rs +41 -0
  223. package/rust/utils/{spinner.rs → src/spinner.rs} +20 -21
  224. package/rust/utils/src/version.rs +27 -0
  225. package/rust/utils/{watcher.rs → src/watcher.rs} +46 -33
  226. package/rust/web/api.rs +5 -0
  227. package/rust/web/cdn.rs +34 -0
  228. package/rust/web/mod.rs +3 -0
  229. package/rust/web/sso.rs +5 -0
  230. package/templates/minimal/README.md +143 -127
  231. package/templates/welcome/README.md +143 -127
  232. package/templates/welcome/src/index.deva +56 -8
  233. package/templates/welcome/src/variables.deva +2 -4
  234. package/tests/integration.rs +21 -0
  235. package/tests/rust/cli_check_build.rs +21 -0
  236. package/tests/rust/cli_help.rs +12 -0
  237. package/tests/rust/cli_template_list.rs +10 -0
  238. package/tests/rust/cli_version.rs +11 -0
  239. package/tests/typescript/index.spec.ts +136 -0
  240. package/tests/typescript/playhead.spec.ts +36 -0
  241. package/tests/typescript/render_e2e.spec.ts +77 -0
  242. package/tsconfig.json +12 -10
  243. package/typescript/bin/index.ts +19 -5
  244. package/typescript/core/functions/index.ts +83 -0
  245. package/typescript/core/index.ts +6 -0
  246. package/typescript/core/types/index.ts +4 -0
  247. package/typescript/core/types/plugin.ts +19 -0
  248. package/typescript/core/types/result.ts +29 -0
  249. package/typescript/core/types/statement.ts +47 -0
  250. package/typescript/core/types/value.ts +29 -0
  251. package/typescript/index.ts +8 -1
  252. package/typescript/pkg/devalang_core.d.ts +4 -0
  253. package/typescript/pkg/devalang_core.ts +49 -0
  254. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  255. package/typescript/scripts/postinstall.ts +85 -0
  256. package/typescript/scripts/version/bump.ts +0 -1
  257. package/typescript/scripts/version/fetch.ts +1 -6
  258. package/typescript/scripts/version/index.ts +0 -1
  259. package/docs/COMMANDS.md +0 -85
  260. package/docs/CONFIG.md +0 -30
  261. package/docs/SYNTAX.md +0 -210
  262. package/out-tsc/bin/devalang.exe +0 -0
  263. package/out-tsc/scripts/postbuild.js +0 -11
  264. package/rust/cli/build.rs +0 -137
  265. package/rust/cli/check.rs +0 -117
  266. package/rust/cli/play.rs +0 -193
  267. package/rust/config/loader.rs +0 -13
  268. package/rust/core/audio/engine.rs +0 -126
  269. package/rust/core/parser/handler/identifier.rs +0 -262
  270. package/rust/core/shared/duration.rs +0 -8
  271. package/rust/core/shared/mod.rs +0 -2
  272. package/rust/core/shared/value.rs +0 -18
  273. package/rust/core/utils/validation.rs +0 -35
  274. package/rust/utils/file.rs +0 -35
  275. package/rust/utils/signature.rs +0 -17
  276. package/rust/utils/version.rs +0 -15
  277. package/typescript/scripts/postbuild.ts +0 -8
@@ -1,145 +1,318 @@
1
- use std::collections::HashMap;
2
-
3
- use crate::{
4
- core::{
5
- parser::statement::{ Statement, StatementKind },
6
- preprocessor::{ module::Module, resolver::trigger::resolve_trigger },
7
- shared::value::Value,
8
- store::global::GlobalStore,
9
- },
10
- utils::logger::Logger,
11
- };
12
-
13
- pub fn resolve_loop(
14
- stmt: &Statement,
15
- module: &Module,
16
- path: &str,
17
- global_store: &GlobalStore
18
- ) -> Statement {
19
- let logger = Logger::new();
20
-
21
- if let Value::Map(value_map) = &stmt.value {
22
- let iterator_value = match value_map.get("iterator") {
23
- Some(Value::Identifier(ident)) => {
24
- match module.variable_table.get(ident) {
25
- Some(Value::Number(n)) => Value::Number(*n),
26
- Some(_) => {
27
- log_type_error(
28
- &logger,
29
- module,
30
- stmt,
31
- format!("Loop iterator '{ident}' must resolve to a number")
32
- );
33
- Value::Null
34
- }
35
- None => {
36
- // Value is not a variable so we assume it's a number
37
- if let Ok(n) = ident.parse::<f32>() {
38
- Value::Number(n)
39
- } else {
40
- log_type_error(
41
- &logger,
42
- module,
43
- stmt,
44
- format!("Loop iterator '{ident}' is not a valid number")
45
- );
46
- Value::Null
47
- }
48
- }
49
- }
50
- }
51
- Some(Value::Number(n)) => Value::Number(*n),
52
- Some(other) => {
53
- log_type_error(
54
- &logger,
55
- module,
56
- stmt,
57
- format!("Unexpected value for loop iterator: {:?}", other)
58
- );
59
- Value::Null
60
- }
61
- None => {
62
- log_type_error(
63
- &logger,
64
- module,
65
- stmt,
66
- "Missing 'iterator' key in loop statement map".to_string()
67
- );
68
- Value::Null
69
- }
70
- };
71
-
72
- let body_value = match value_map.get("body") {
73
- Some(Value::Block(block)) => {
74
- let mut resolved_block = Vec::new();
75
- for ref statement in block.clone() {
76
- match &statement.kind {
77
- StatementKind::Trigger { entity, duration } => {
78
- let resolved = resolve_trigger(
79
- &mut statement.clone(),
80
- &entity,
81
- &mut duration.clone(),
82
- module,
83
- path,
84
- global_store
85
- );
86
- resolved_block.push(resolved);
87
- }
88
- _ => {
89
- println!("Unhandled loop body statement: {:?}", statement);
90
- }
91
- }
92
- }
93
- Value::Block(resolved_block)
94
- }
95
- Some(other) => {
96
- log_type_error(
97
- &logger,
98
- module,
99
- stmt,
100
- format!("Unexpected value for loop body: {:?}", other)
101
- );
102
- Value::Null
103
- }
104
- None => {
105
- log_type_error(
106
- &logger,
107
- module,
108
- stmt,
109
- "Missing 'body' key in loop statement map".to_string()
110
- );
111
- Value::Null
112
- }
113
- };
114
-
115
- let mut resolved_map = HashMap::new();
116
- resolved_map.insert("iterator".to_string(), iterator_value);
117
- resolved_map.insert("body".to_string(), body_value);
118
-
119
- Statement {
120
- kind: StatementKind::Loop,
121
- value: Value::Map(resolved_map),
122
- ..stmt.clone()
123
- }
124
- } else {
125
- log_type_error(
126
- &logger,
127
- module,
128
- stmt,
129
- format!("Expected a map for loop value, found {:?}", stmt.value)
130
- );
131
-
132
- Statement {
133
- kind: StatementKind::Error {
134
- message: "Expected a map for loop value".to_string(),
135
- },
136
- value: Value::Null,
137
- ..stmt.clone()
138
- }
139
- }
140
- }
141
-
142
- fn log_type_error(logger: &Logger, module: &Module, stmt: &Statement, message: String) {
143
- let stacktrace = format!("{}:{}:{}", module.path, stmt.line, stmt.column);
144
- logger.log_error_with_stacktrace(&message, &stacktrace);
145
- }
1
+ use std::collections::HashMap;
2
+
3
+ use crate::core::{
4
+ parser::statement::{Statement, StatementKind},
5
+ preprocessor::{
6
+ module::Module,
7
+ resolver::{driver::resolve_statement, value::resolve_value},
8
+ },
9
+ store::global::GlobalStore,
10
+ };
11
+ use devalang_types::Value;
12
+ use devalang_utils::logger::Logger;
13
+
14
+ pub fn resolve_loop(
15
+ stmt: &Statement,
16
+ module: &Module,
17
+ path: &str,
18
+ global_store: &mut GlobalStore,
19
+ ) -> Statement {
20
+ let logger = Logger::new();
21
+
22
+ let resolved_value = resolve_value(&stmt.value, module, global_store);
23
+
24
+ let Value::Map(value_map) = &resolved_value else {
25
+ return error_stmt(&logger, module, stmt, "Expected a map for loop value");
26
+ };
27
+
28
+ let mut resolved_map: HashMap<String, Value> = HashMap::new();
29
+ for (key, val) in value_map {
30
+ resolved_map.insert(key.clone(), resolve_value(val, module, global_store));
31
+ }
32
+
33
+ // Foreach form takes precedence if present
34
+ if let (Some(Value::Identifier(var_name)), Some(array_val)) =
35
+ (resolved_map.get("foreach"), resolved_map.get("array"))
36
+ {
37
+ // Normalize array_val into an iterable Array
38
+ let resolved_array = match array_val {
39
+ Value::Array(items) => Value::Array(
40
+ items
41
+ .iter()
42
+ .map(|v| resolve_value(v, module, global_store))
43
+ .collect(),
44
+ ),
45
+ Value::Number(n) => {
46
+ // Iterate 0..n-1
47
+ let count = (*n).max(0.0) as usize;
48
+ let mut items = Vec::with_capacity(count);
49
+ for i in 0..count {
50
+ items.push(Value::Number(i as f32));
51
+ }
52
+ Value::Array(items)
53
+ }
54
+ Value::String(s) => {
55
+ // Try to parse a simple comma-separated list: "a,b,c" -> ["a","b","c"]
56
+ // If numeric string: iterate 0..n-1
57
+ if let Ok(n) = s.parse::<f32>() {
58
+ let count = n.max(0.0) as usize;
59
+ let mut items = Vec::with_capacity(count);
60
+ for i in 0..count {
61
+ items.push(Value::Number(i as f32));
62
+ }
63
+ Value::Array(items)
64
+ } else if s.contains(',') {
65
+ let parts: Vec<Value> = s
66
+ .split(',')
67
+ .map(|p| Value::String(p.trim().to_string()))
68
+ .collect();
69
+ Value::Array(parts)
70
+ } else {
71
+ // Fallback: iterate characters
72
+ let parts: Vec<Value> =
73
+ s.chars().map(|c| Value::String(c.to_string())).collect();
74
+ Value::Array(parts)
75
+ }
76
+ }
77
+ Value::Identifier(name) => {
78
+ // Resolve identifier from module variables (already resolved map above)
79
+ let v = if let Some(v) = module.variable_table.get(name) {
80
+ v.clone()
81
+ } else {
82
+ Value::Null
83
+ };
84
+ match v {
85
+ Value::Array(items) => Value::Array(
86
+ items
87
+ .iter()
88
+ .map(|v| resolve_value(v, module, global_store))
89
+ .collect(),
90
+ ),
91
+ Value::Number(n) => {
92
+ let count = n.max(0.0) as usize;
93
+ let mut items = Vec::with_capacity(count);
94
+ for i in 0..count {
95
+ items.push(Value::Number(i as f32));
96
+ }
97
+ Value::Array(items)
98
+ }
99
+ Value::String(s) => {
100
+ if let Ok(n) = s.parse::<f32>() {
101
+ let count = n.max(0.0) as usize;
102
+ let mut items = Vec::with_capacity(count);
103
+ for i in 0..count {
104
+ items.push(Value::Number(i as f32));
105
+ }
106
+ Value::Array(items)
107
+ } else if s.contains(',') {
108
+ let parts: Vec<Value> = s
109
+ .split(',')
110
+ .map(|p| Value::String(p.trim().to_string()))
111
+ .collect();
112
+ Value::Array(parts)
113
+ } else {
114
+ let parts: Vec<Value> =
115
+ s.chars().map(|c| Value::String(c.to_string())).collect();
116
+ Value::Array(parts)
117
+ }
118
+ }
119
+ other => {
120
+ error_value(
121
+ &logger,
122
+ module,
123
+ stmt,
124
+ &format!(
125
+ "Foreach identifier '{}' resolves to unsupported value: {:?}",
126
+ name, other
127
+ ),
128
+ );
129
+ Value::Array(vec![])
130
+ }
131
+ }
132
+ }
133
+ other => {
134
+ // Resolve and normalize if possible
135
+ let v = resolve_value(other, module, global_store);
136
+ match v {
137
+ Value::Array(items) => Value::Array(items),
138
+ Value::Number(n) => {
139
+ let count = n.max(0.0) as usize;
140
+ let mut items = Vec::with_capacity(count);
141
+ for i in 0..count {
142
+ items.push(Value::Number(i as f32));
143
+ }
144
+ Value::Array(items)
145
+ }
146
+ Value::String(s) => {
147
+ if let Ok(n) = s.parse::<f32>() {
148
+ let count = n.max(0.0) as usize;
149
+ let mut items = Vec::with_capacity(count);
150
+ for i in 0..count {
151
+ items.push(Value::Number(i as f32));
152
+ }
153
+ Value::Array(items)
154
+ } else if s.contains(',') {
155
+ let parts: Vec<Value> = s
156
+ .split(',')
157
+ .map(|p| Value::String(p.trim().to_string()))
158
+ .collect();
159
+ Value::Array(parts)
160
+ } else {
161
+ let parts: Vec<Value> =
162
+ s.chars().map(|c| Value::String(c.to_string())).collect();
163
+ Value::Array(parts)
164
+ }
165
+ }
166
+ other => {
167
+ error_value(
168
+ &logger,
169
+ module,
170
+ stmt,
171
+ &format!("Unsupported foreach array value: {:?}", other),
172
+ );
173
+ Value::Array(vec![])
174
+ }
175
+ }
176
+ }
177
+ };
178
+
179
+ let body_value = match resolved_map.get("body") {
180
+ Some(Value::Block(stmts)) => {
181
+ let resolved = stmts
182
+ .iter()
183
+ .map(|s| resolve_statement(s, module, path, global_store))
184
+ .collect();
185
+ Value::Block(resolved)
186
+ }
187
+ _ => {
188
+ error_value(&logger, module, stmt, "Invalid or missing loop body");
189
+ Value::Block(vec![])
190
+ }
191
+ };
192
+
193
+ let mut final_map = HashMap::new();
194
+ final_map.insert("foreach".to_string(), Value::Identifier(var_name.clone()));
195
+ final_map.insert("array".to_string(), resolved_array);
196
+ final_map.insert("body".to_string(), body_value);
197
+
198
+ return Statement {
199
+ kind: StatementKind::Loop,
200
+ value: Value::Map(final_map),
201
+ ..stmt.clone()
202
+ };
203
+ }
204
+
205
+ let iterator_value = match resolved_map.get("iterator") {
206
+ Some(Value::Number(n)) => Value::Number(*n),
207
+ Some(Value::String(s)) => {
208
+ if let Ok(n) = s.parse::<f32>() {
209
+ Value::Number(n)
210
+ } else {
211
+ error_value(
212
+ &logger,
213
+ module,
214
+ stmt,
215
+ &format!("Loop iterator string not numeric: '{}'", s),
216
+ );
217
+ Value::Number(1.0)
218
+ }
219
+ }
220
+ Some(Value::Identifier(name)) => {
221
+ // Try resolving from module vars (may be number or numeric string)
222
+ if let Some(v) = module.variable_table.get(name) {
223
+ match v {
224
+ Value::Number(n) => Value::Number(*n),
225
+ Value::String(s) => {
226
+ if let Ok(n) = s.parse::<f32>() {
227
+ Value::Number(n)
228
+ } else {
229
+ error_value(
230
+ &logger,
231
+ module,
232
+ stmt,
233
+ &format!(
234
+ "Loop iterator '{}' resolves to non-numeric string: '{}'",
235
+ name, s
236
+ ),
237
+ );
238
+ Value::Number(1.0)
239
+ }
240
+ }
241
+ other => {
242
+ error_value(
243
+ &logger,
244
+ module,
245
+ stmt,
246
+ &format!(
247
+ "Loop iterator '{}' resolves to non-number: {:?}",
248
+ name, other
249
+ ),
250
+ );
251
+ Value::Number(1.0)
252
+ }
253
+ }
254
+ } else {
255
+ error_value(
256
+ &logger,
257
+ module,
258
+ stmt,
259
+ &format!("Loop iterator identifier '{}' not found", name),
260
+ );
261
+ Value::Number(1.0)
262
+ }
263
+ }
264
+ Some(other) => {
265
+ error_value(
266
+ &logger,
267
+ module,
268
+ stmt,
269
+ &format!("Loop iterator must be a number, found: {:?}", other),
270
+ );
271
+ Value::Number(1.0)
272
+ }
273
+ None => {
274
+ error_value(&logger, module, stmt, "Missing 'iterator' in loop");
275
+ Value::Number(1.0)
276
+ }
277
+ };
278
+
279
+ let body_value = match resolved_map.get("body") {
280
+ Some(Value::Block(stmts)) => {
281
+ let resolved = stmts
282
+ .iter()
283
+ .map(|s| resolve_statement(s, module, path, global_store))
284
+ .collect();
285
+ Value::Block(resolved)
286
+ }
287
+ _ => {
288
+ error_value(&logger, module, stmt, "Invalid or missing loop body");
289
+ Value::Block(vec![])
290
+ }
291
+ };
292
+
293
+ let mut final_map = HashMap::new();
294
+ final_map.insert("iterator".to_string(), iterator_value);
295
+ final_map.insert("body".to_string(), body_value);
296
+
297
+ Statement {
298
+ kind: StatementKind::Loop,
299
+ value: Value::Map(final_map),
300
+ ..stmt.clone()
301
+ }
302
+ }
303
+
304
+ fn error_value(logger: &Logger, module: &Module, stmt: &Statement, msg: &str) {
305
+ let stacktrace = format!("{}:{}:{}", module.path, stmt.line, stmt.column);
306
+ logger.log_error_with_stacktrace(msg, &stacktrace);
307
+ }
308
+
309
+ fn error_stmt(logger: &Logger, module: &Module, stmt: &Statement, msg: &str) -> Statement {
310
+ error_value(logger, module, stmt, msg);
311
+ Statement {
312
+ kind: StatementKind::Error {
313
+ message: msg.to_string(),
314
+ },
315
+ value: Value::Null,
316
+ ..stmt.clone()
317
+ }
318
+ }
@@ -1,10 +1,16 @@
1
- pub mod driver;
2
-
3
- pub mod trigger;
4
- pub mod loop_;
5
- pub mod bank;
6
- pub mod tempo;
7
- pub mod group;
8
- pub mod condition;
9
- pub mod spawn;
10
- pub mod call;
1
+ pub mod driver;
2
+
3
+ pub mod value;
4
+
5
+ pub mod bank;
6
+ pub mod call;
7
+ pub mod condition;
8
+ pub mod function;
9
+ pub mod group;
10
+ pub mod let_;
11
+ pub mod loop_;
12
+ pub mod pattern;
13
+ pub mod spawn;
14
+ pub mod synth;
15
+ pub mod tempo;
16
+ pub mod trigger;
@@ -0,0 +1,83 @@
1
+ use crate::core::{
2
+ parser::statement::{Statement, StatementKind},
3
+ preprocessor::module::Module,
4
+ store::global::GlobalStore,
5
+ };
6
+ use devalang_types::Value;
7
+ use devalang_utils::logger::{LogLevel, Logger};
8
+
9
+ pub fn resolve_pattern(
10
+ stmt: &Statement,
11
+ module: &Module,
12
+ path: &str,
13
+ global_store: &mut GlobalStore,
14
+ ) -> Statement {
15
+ let logger = Logger::new();
16
+
17
+ // Expecting pattern name stored on the Statement.kind; value may contain the string
18
+ if let StatementKind::Pattern { name, target } = &stmt.kind {
19
+ // Ensure name doesn't already exist
20
+ if global_store.variables.variables.contains_key(name) {
21
+ logger.log_error_with_stacktrace(
22
+ &format!("Pattern identifier '{}' already exists", name),
23
+ path,
24
+ );
25
+ return Statement {
26
+ kind: StatementKind::Error {
27
+ message: format!("Pattern '{}' already exists", name),
28
+ },
29
+ ..stmt.clone()
30
+ };
31
+ }
32
+
33
+ // Resolve potential target and pattern string value
34
+ let resolved_value = resolve_value(&stmt.value, module, global_store);
35
+
36
+ // Build a map to store the pattern definition
37
+ let mut map = std::collections::HashMap::new();
38
+ map.insert("identifier".to_string(), Value::String(name.clone()));
39
+ if let Some(t) = target {
40
+ map.insert("target".to_string(), Value::String(t.clone()));
41
+ }
42
+ // Keep raw pattern in 'pattern' key
43
+ map.insert("pattern".to_string(), resolved_value.clone());
44
+
45
+ let resolved_stmt = Statement {
46
+ kind: StatementKind::Pattern {
47
+ name: name.clone(),
48
+ target: target.clone(),
49
+ },
50
+ value: resolved_value,
51
+ ..stmt.clone()
52
+ };
53
+
54
+ // Store into global variables as a Statement
55
+ global_store.variables.variables.insert(
56
+ name.clone(),
57
+ Value::Statement(Box::new(resolved_stmt.clone())),
58
+ );
59
+
60
+ return resolved_stmt;
61
+ }
62
+
63
+ logger.log_message(
64
+ LogLevel::Warning,
65
+ "resolve_pattern called on non-pattern statement",
66
+ );
67
+ stmt.clone()
68
+ }
69
+
70
+ fn resolve_value(value: &Value, module: &Module, global_store: &mut GlobalStore) -> Value {
71
+ // reuse driver::resolve_value logic; simple local resolution for pattern value
72
+ match value {
73
+ Value::String(s) => Value::String(s.clone()),
74
+ Value::Map(m) => {
75
+ let mut resolved = std::collections::HashMap::new();
76
+ for (k, v) in m {
77
+ resolved.insert(k.clone(), resolve_value(v, module, global_store));
78
+ }
79
+ Value::Map(resolved)
80
+ }
81
+ other => other.clone(),
82
+ }
83
+ }