@devaloop/devalang 0.0.1-alpha.9 → 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 (271) 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 -154
  6. package/docs/CHANGELOG.md +386 -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/duration.deva +9 -0
  13. package/examples/events.deva +12 -0
  14. package/examples/function.deva +15 -0
  15. package/examples/index.deva +57 -12
  16. package/examples/loop.deva +5 -12
  17. package/examples/pattern.deva +8 -0
  18. package/examples/plugin.deva +16 -0
  19. package/examples/variables.deva +1 -1
  20. package/out-tsc/bin/index.d.ts +2 -0
  21. package/out-tsc/bin/index.js +51 -7
  22. package/out-tsc/core/functions/index.d.ts +37 -0
  23. package/out-tsc/core/functions/index.js +76 -0
  24. package/out-tsc/core/index.d.ts +6 -0
  25. package/out-tsc/core/index.js +22 -0
  26. package/out-tsc/core/types/index.d.ts +4 -0
  27. package/out-tsc/core/types/index.js +20 -0
  28. package/out-tsc/core/types/plugin.d.ts +18 -0
  29. package/out-tsc/core/types/plugin.js +2 -0
  30. package/out-tsc/core/types/result.d.ts +27 -0
  31. package/out-tsc/core/types/result.js +2 -0
  32. package/out-tsc/core/types/statement.d.ts +106 -0
  33. package/out-tsc/core/types/statement.js +2 -0
  34. package/out-tsc/core/types/value.d.ts +43 -0
  35. package/out-tsc/core/types/value.js +2 -0
  36. package/out-tsc/index.d.ts +7 -0
  37. package/out-tsc/index.js +42 -1
  38. package/out-tsc/pkg/devalang_core.d.ts +13 -0
  39. package/out-tsc/pkg/devalang_core.js +50 -0
  40. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +33 -0
  41. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  42. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  43. package/out-tsc/scripts/postinstall.d.ts +1 -0
  44. package/out-tsc/scripts/postinstall.js +83 -0
  45. package/out-tsc/scripts/version/bump.d.ts +1 -0
  46. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  47. package/out-tsc/scripts/version/index.d.ts +1 -0
  48. package/out-tsc/scripts/version/sync.d.ts +1 -0
  49. package/package.json +28 -7
  50. package/project-version.json +4 -4
  51. package/rust/cli/bank/api.rs +122 -0
  52. package/rust/cli/bank/commands.rs +275 -0
  53. package/rust/cli/bank/mod.rs +29 -0
  54. package/rust/cli/build/commands.rs +103 -0
  55. package/rust/cli/build/mod.rs +2 -0
  56. package/rust/cli/build/process.rs +146 -0
  57. package/rust/cli/check/mod.rs +208 -0
  58. package/rust/cli/discover/commands.rs +253 -0
  59. package/rust/cli/discover/config.rs +111 -0
  60. package/rust/cli/discover/fs.rs +19 -0
  61. package/rust/cli/discover/install.rs +103 -0
  62. package/rust/cli/discover/metadata.rs +48 -0
  63. package/rust/cli/discover/mod.rs +5 -0
  64. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  65. package/rust/cli/init/mod.rs +1 -0
  66. package/rust/cli/install/addon.rs +118 -0
  67. package/rust/cli/install/bank.rs +53 -0
  68. package/rust/cli/install/commands.rs +35 -0
  69. package/rust/cli/install/mod.rs +4 -0
  70. package/rust/cli/install/plugin.rs +61 -0
  71. package/rust/cli/login/commands.rs +124 -0
  72. package/rust/cli/login/mod.rs +1 -0
  73. package/rust/cli/mod.rs +12 -205
  74. package/rust/cli/parser.rs +314 -0
  75. package/rust/cli/play/commands.rs +324 -0
  76. package/rust/cli/play/io.rs +17 -0
  77. package/rust/cli/play/mod.rs +5 -0
  78. package/rust/cli/play/process.rs +150 -0
  79. package/rust/cli/play/realtime.rs +91 -0
  80. package/rust/cli/play/utils.rs +23 -0
  81. package/rust/cli/telemetry/commands.rs +22 -0
  82. package/rust/cli/telemetry/event_creator.rs +80 -0
  83. package/rust/cli/telemetry/mod.rs +3 -0
  84. package/rust/cli/telemetry/send.rs +51 -0
  85. package/rust/cli/{template.rs → template/commands.rs} +69 -57
  86. package/rust/cli/template/mod.rs +1 -0
  87. package/rust/cli/update/commands.rs +6 -0
  88. package/rust/cli/update/mod.rs +1 -0
  89. package/rust/config/driver.rs +103 -0
  90. package/rust/config/mod.rs +3 -16
  91. package/rust/config/ops.rs +26 -0
  92. package/rust/config/settings.rs +101 -0
  93. package/rust/core/audio/engine/helpers.rs +170 -0
  94. package/rust/core/audio/engine/mod.rs +7 -0
  95. package/rust/core/audio/engine/sample.rs +366 -0
  96. package/rust/core/audio/engine/synth.rs +325 -0
  97. package/rust/core/audio/evaluator.rs +310 -31
  98. package/rust/core/audio/interpreter/arrow_call.rs +311 -129
  99. package/rust/core/audio/interpreter/automate.rs +18 -0
  100. package/rust/core/audio/interpreter/call.rs +294 -64
  101. package/rust/core/audio/interpreter/condition.rs +71 -69
  102. package/rust/core/audio/interpreter/driver.rs +542 -216
  103. package/rust/core/audio/interpreter/function.rs +26 -0
  104. package/rust/core/audio/interpreter/let_.rs +38 -19
  105. package/rust/core/audio/interpreter/load.rs +19 -18
  106. package/rust/core/audio/interpreter/loop_.rs +114 -67
  107. package/rust/core/audio/interpreter/mod.rs +14 -12
  108. package/rust/core/audio/interpreter/sleep.rs +28 -36
  109. package/rust/core/audio/interpreter/spawn.rs +252 -66
  110. package/rust/core/audio/interpreter/tempo.rs +40 -16
  111. package/rust/core/audio/interpreter/trigger.rs +239 -69
  112. package/rust/core/audio/loader/mod.rs +1 -1
  113. package/rust/core/audio/loader/trigger.rs +97 -52
  114. package/rust/core/audio/mod.rs +7 -6
  115. package/rust/core/audio/player.rs +70 -54
  116. package/rust/core/audio/renderer.rs +54 -54
  117. package/rust/core/audio/special/easing.rs +189 -0
  118. package/rust/core/audio/special/env.rs +45 -0
  119. package/rust/core/audio/special/math.rs +134 -0
  120. package/rust/core/audio/special/mod.rs +9 -0
  121. package/rust/core/audio/special/modulator.rs +143 -0
  122. package/rust/core/builder/mod.rs +86 -80
  123. package/rust/core/debugger/lexer.rs +27 -27
  124. package/rust/core/debugger/mod.rs +30 -21
  125. package/rust/core/debugger/module.rs +55 -0
  126. package/rust/core/debugger/preprocessor.rs +27 -27
  127. package/rust/core/debugger/store.rs +40 -25
  128. package/rust/core/error/mod.rs +269 -60
  129. package/rust/core/lexer/driver.rs +61 -0
  130. package/rust/core/lexer/handler/arrow.rs +82 -31
  131. package/rust/core/lexer/handler/at.rs +21 -21
  132. package/rust/core/lexer/handler/brace.rs +41 -41
  133. package/rust/core/lexer/handler/colon.rs +21 -21
  134. package/rust/core/lexer/handler/comment.rs +30 -30
  135. package/rust/core/lexer/handler/dot.rs +21 -21
  136. package/rust/core/lexer/handler/driver.rs +337 -226
  137. package/rust/core/lexer/handler/identifier.rs +47 -41
  138. package/rust/core/lexer/handler/indent.rs +66 -52
  139. package/rust/core/lexer/handler/mod.rs +15 -14
  140. package/rust/core/lexer/handler/newline.rs +23 -23
  141. package/rust/core/lexer/handler/number.rs +31 -31
  142. package/rust/core/lexer/handler/operator.rs +46 -44
  143. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  144. package/rust/core/lexer/handler/slash.rs +21 -0
  145. package/rust/core/lexer/handler/string.rs +63 -63
  146. package/rust/core/lexer/mod.rs +3 -51
  147. package/rust/core/lexer/token.rs +17 -12
  148. package/rust/core/mod.rs +10 -10
  149. package/rust/core/parser/driver.rs +584 -331
  150. package/rust/core/parser/handler/arrow_call.rs +253 -126
  151. package/rust/core/parser/handler/at.rs +279 -162
  152. package/rust/core/parser/handler/bank.rs +104 -41
  153. package/rust/core/parser/handler/condition.rs +83 -74
  154. package/rust/core/parser/handler/dot.rs +148 -112
  155. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  156. package/rust/core/parser/handler/identifier/call.rs +91 -41
  157. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  158. package/rust/core/parser/handler/identifier/function.rs +113 -0
  159. package/rust/core/parser/handler/identifier/group.rs +89 -75
  160. package/rust/core/parser/handler/identifier/let_.rs +173 -133
  161. package/rust/core/parser/handler/identifier/mod.rs +55 -51
  162. package/rust/core/parser/handler/identifier/on.rs +107 -0
  163. package/rust/core/parser/handler/identifier/print.rs +49 -0
  164. package/rust/core/parser/handler/identifier/sleep.rs +43 -33
  165. package/rust/core/parser/handler/identifier/spawn.rs +91 -41
  166. package/rust/core/parser/handler/identifier/synth.rs +135 -65
  167. package/rust/core/parser/handler/loop_.rs +194 -72
  168. package/rust/core/parser/handler/mod.rs +9 -8
  169. package/rust/core/parser/handler/pattern.rs +74 -0
  170. package/rust/core/parser/handler/tempo.rs +57 -47
  171. package/rust/core/parser/mod.rs +3 -4
  172. package/rust/core/parser/statement.rs +11 -96
  173. package/rust/core/plugin/loader.rs +137 -0
  174. package/rust/core/plugin/mod.rs +2 -0
  175. package/rust/core/plugin/runner.rs +347 -0
  176. package/rust/core/preprocessor/loader.rs +637 -193
  177. package/rust/core/preprocessor/mod.rs +4 -4
  178. package/rust/core/preprocessor/module.rs +60 -50
  179. package/rust/core/preprocessor/processor.rs +114 -76
  180. package/rust/core/preprocessor/resolver/bank.rs +49 -47
  181. package/rust/core/preprocessor/resolver/call.rs +124 -123
  182. package/rust/core/preprocessor/resolver/condition.rs +95 -92
  183. package/rust/core/preprocessor/resolver/driver.rs +324 -227
  184. package/rust/core/preprocessor/resolver/function.rs +69 -0
  185. package/rust/core/preprocessor/resolver/group.rs +94 -61
  186. package/rust/core/preprocessor/resolver/let_.rs +32 -31
  187. package/rust/core/preprocessor/resolver/loop_.rs +318 -91
  188. package/rust/core/preprocessor/resolver/mod.rs +16 -14
  189. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  190. package/rust/core/preprocessor/resolver/spawn.rs +99 -58
  191. package/rust/core/preprocessor/resolver/synth.rs +54 -50
  192. package/rust/core/preprocessor/resolver/tempo.rs +48 -49
  193. package/rust/core/preprocessor/resolver/trigger.rs +116 -112
  194. package/rust/core/preprocessor/resolver/value.rs +176 -78
  195. package/rust/core/store/export.rs +28 -28
  196. package/rust/core/store/function.rs +40 -0
  197. package/rust/core/store/global.rs +61 -39
  198. package/rust/core/store/import.rs +28 -28
  199. package/rust/core/store/mod.rs +5 -4
  200. package/rust/core/store/variable.rs +51 -28
  201. package/rust/core/utils/mod.rs +1 -2
  202. package/rust/core/utils/path.rs +37 -31
  203. package/rust/lib.rs +308 -117
  204. package/rust/main.rs +364 -65
  205. package/rust/types/Cargo.toml +11 -0
  206. package/rust/types/src/addons.rs +55 -0
  207. package/rust/types/src/ast.rs +202 -0
  208. package/rust/types/src/config.rs +74 -0
  209. package/rust/types/src/lib.rs +12 -0
  210. package/rust/types/src/telemetry.rs +85 -0
  211. package/rust/utils/Cargo.toml +26 -0
  212. package/rust/utils/src/error.rs +186 -0
  213. package/rust/utils/src/file.rs +94 -0
  214. package/rust/utils/src/first_usage.rs +97 -0
  215. package/rust/utils/{mod.rs → src/lib.rs} +9 -6
  216. package/rust/utils/{logger.rs → src/logger.rs} +200 -123
  217. package/rust/utils/src/path.rs +88 -0
  218. package/rust/utils/src/signature.rs +41 -0
  219. package/rust/utils/{spinner.rs → src/spinner.rs} +20 -21
  220. package/rust/utils/src/version.rs +27 -0
  221. package/rust/utils/{watcher.rs → src/watcher.rs} +46 -33
  222. package/rust/web/api.rs +5 -0
  223. package/rust/web/cdn.rs +34 -0
  224. package/rust/web/mod.rs +3 -0
  225. package/rust/web/sso.rs +5 -0
  226. package/templates/minimal/README.md +143 -127
  227. package/templates/welcome/README.md +143 -127
  228. package/templates/welcome/src/index.deva +56 -8
  229. package/templates/welcome/src/variables.deva +2 -4
  230. package/tests/integration.rs +21 -0
  231. package/tests/rust/cli_check_build.rs +21 -0
  232. package/tests/rust/cli_help.rs +12 -0
  233. package/tests/rust/cli_template_list.rs +10 -0
  234. package/tests/rust/cli_version.rs +11 -0
  235. package/tests/typescript/index.spec.ts +136 -0
  236. package/tests/typescript/playhead.spec.ts +36 -0
  237. package/tests/typescript/render_e2e.spec.ts +77 -0
  238. package/tsconfig.json +12 -10
  239. package/typescript/bin/index.ts +19 -5
  240. package/typescript/core/functions/index.ts +83 -0
  241. package/typescript/core/index.ts +6 -0
  242. package/typescript/core/types/index.ts +4 -0
  243. package/typescript/core/types/plugin.ts +19 -0
  244. package/typescript/core/types/result.ts +29 -0
  245. package/typescript/core/types/statement.ts +47 -0
  246. package/typescript/core/types/value.ts +29 -0
  247. package/typescript/index.ts +8 -1
  248. package/typescript/pkg/devalang_core.d.ts +4 -0
  249. package/typescript/pkg/devalang_core.ts +49 -0
  250. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  251. package/typescript/scripts/postinstall.ts +85 -0
  252. package/typescript/scripts/version/bump.ts +0 -1
  253. package/typescript/scripts/version/index.ts +0 -1
  254. package/docs/COMMANDS.md +0 -85
  255. package/docs/CONFIG.md +0 -30
  256. package/docs/SYNTAX.md +0 -210
  257. package/out-tsc/bin/devalang.exe +0 -0
  258. package/out-tsc/scripts/postbuild.js +0 -11
  259. package/rust/cli/build.rs +0 -137
  260. package/rust/cli/check.rs +0 -117
  261. package/rust/cli/play.rs +0 -193
  262. package/rust/config/loader.rs +0 -13
  263. package/rust/core/audio/engine.rs +0 -203
  264. package/rust/core/shared/duration.rs +0 -8
  265. package/rust/core/shared/mod.rs +0 -2
  266. package/rust/core/shared/value.rs +0 -18
  267. package/rust/core/utils/validation.rs +0 -37
  268. package/rust/utils/file.rs +0 -35
  269. package/rust/utils/signature.rs +0 -17
  270. package/rust/utils/version.rs +0 -15
  271. package/typescript/scripts/postbuild.ts +0 -8
@@ -0,0 +1,254 @@
1
+ use crate::core::{
2
+ lexer::token::{Token, TokenKind},
3
+ parser::{
4
+ driver::Parser,
5
+ statement::{Statement, StatementKind},
6
+ },
7
+ store::global::GlobalStore,
8
+ };
9
+ use devalang_types::Value;
10
+ use std::collections::HashMap;
11
+
12
+ pub fn parse_automate_token(
13
+ parser: &mut Parser,
14
+ current_token: Token,
15
+ _global_store: &mut GlobalStore,
16
+ ) -> Statement {
17
+ parser.advance(); // consume 'automate'
18
+
19
+ // Expect target identifier
20
+ let Some(target_token) = parser.peek_clone() else {
21
+ return crate::core::parser::statement::error_from_token(
22
+ current_token,
23
+ "Expected target after 'automate'".to_string(),
24
+ );
25
+ };
26
+
27
+ if target_token.kind != TokenKind::Identifier && target_token.kind != TokenKind::String {
28
+ return crate::core::parser::statement::error_from_token(
29
+ target_token,
30
+ "Expected valid target after 'automate'".to_string(),
31
+ );
32
+ }
33
+ parser.advance(); // consume target
34
+
35
+ // Expect ':'
36
+ let Some(colon_token) = parser.peek_clone() else {
37
+ return crate::core::parser::statement::error_from_token(
38
+ target_token,
39
+ "Expected ':' after automate target".to_string(),
40
+ );
41
+ };
42
+ if colon_token.kind != TokenKind::Colon {
43
+ return crate::core::parser::statement::error_from_token(
44
+ colon_token,
45
+ "Expected ':' after automate target".to_string(),
46
+ );
47
+ }
48
+ parser.advance(); // consume ':'
49
+
50
+ let base_indent = current_token.indent;
51
+
52
+ // Collect tokens inside block (indented > base_indent)
53
+ let mut index = parser.token_index;
54
+ let mut tokens_inside = Vec::new();
55
+ while index < parser.tokens.len() {
56
+ let tok = parser.tokens[index].clone();
57
+ if tok.indent <= base_indent && tok.kind != TokenKind::Newline {
58
+ break;
59
+ }
60
+ tokens_inside.push(tok);
61
+ index += 1;
62
+ }
63
+ parser.token_index = index;
64
+
65
+ // Now parse block manually to capture 'param' entries without reusing general parser kinds
66
+ let mut local = Parser {
67
+ resolve_modules: parser.resolve_modules,
68
+ tokens: tokens_inside,
69
+ token_index: 0,
70
+ current_module: parser.current_module.clone(),
71
+ previous: None,
72
+ };
73
+
74
+ let mut params: HashMap<String, Value> = HashMap::new();
75
+
76
+ while let Some(tok) = local.peek_clone() {
77
+ match tok.kind {
78
+ TokenKind::Identifier if tok.lexeme == "param" => {
79
+ local.advance(); // consume 'param'
80
+ // param name
81
+ let Some(name_tok) = local.peek_clone() else {
82
+ return crate::core::parser::statement::error_from_token(
83
+ tok,
84
+ "Expected parameter name after 'param'".to_string(),
85
+ );
86
+ };
87
+ if name_tok.kind != TokenKind::Identifier && name_tok.kind != TokenKind::String {
88
+ return crate::core::parser::statement::error_from_token(
89
+ name_tok,
90
+ "Expected valid parameter name".to_string(),
91
+ );
92
+ }
93
+ local.advance(); // consume name
94
+
95
+ // Expect '{'
96
+ if !local.match_token(TokenKind::LBrace) {
97
+ return crate::core::parser::statement::error_from_token(
98
+ name_tok,
99
+ "Expected '{' to start parameter block".to_string(),
100
+ );
101
+ }
102
+
103
+ // Collect entries like: 0% = 0.0
104
+ let mut envelope: HashMap<String, Value> = HashMap::new();
105
+ while let Some(inner) = local.peek_clone() {
106
+ if inner.kind == TokenKind::RBrace {
107
+ local.advance();
108
+ break;
109
+ }
110
+ // Skip formatting tokens inside the param block
111
+ if matches!(
112
+ inner.kind,
113
+ TokenKind::Newline
114
+ | TokenKind::Indent
115
+ | TokenKind::Dedent
116
+ | TokenKind::Comma
117
+ ) {
118
+ local.advance();
119
+ continue;
120
+ }
121
+
122
+ // Read percentage token: could be number followed by '%' as Dot or Identifier? '%' not defined.
123
+ // Our lexer has no Percent token, so accept either Number or Identifier containing e.g. '0%'.
124
+ let percent_token = inner.clone();
125
+ local.advance();
126
+
127
+ let percent_key = percent_token.lexeme.clone();
128
+
129
+ // Expect '='
130
+ // Skip any stray formatting between key and '='
131
+ while let Some(t) = local.peek_kind() {
132
+ if matches!(
133
+ t,
134
+ TokenKind::Indent | TokenKind::Dedent | TokenKind::Newline
135
+ ) {
136
+ local.advance();
137
+ continue;
138
+ }
139
+ break;
140
+ }
141
+ if !local.match_token(TokenKind::Equals) {
142
+ return crate::core::parser::statement::error_from_token(
143
+ percent_token,
144
+ "Expected '=' in param entry".to_string(),
145
+ );
146
+ }
147
+
148
+ // Read value (number or identifier)
149
+ // Skip formatting before value
150
+ while let Some(t) = local.peek_kind() {
151
+ if matches!(
152
+ t,
153
+ TokenKind::Indent | TokenKind::Dedent | TokenKind::Newline
154
+ ) {
155
+ local.advance();
156
+ continue;
157
+ }
158
+ break;
159
+ }
160
+
161
+ let value = if let Some(vtok) = local.peek_clone() {
162
+ match vtok.kind {
163
+ // Handle negative numbers where '-' is lexed as Arrow
164
+ TokenKind::Arrow => {
165
+ // Check if next token is a number
166
+ let mut num_str = String::from("-");
167
+ local.advance(); // consume '-'
168
+ if let Some(ntok) = local.peek_clone() {
169
+ if ntok.kind == TokenKind::Number {
170
+ num_str.push_str(&ntok.lexeme);
171
+ local.advance(); // consume number
172
+ if let Some(dot) = local.peek_clone() {
173
+ if dot.kind == TokenKind::Dot {
174
+ local.advance();
175
+ if let Some(frac) = local.peek_clone() {
176
+ if frac.kind == TokenKind::Number {
177
+ num_str.push('.');
178
+ num_str.push_str(&frac.lexeme);
179
+ local.advance();
180
+ }
181
+ }
182
+ }
183
+ }
184
+ Value::Number(num_str.parse::<f32>().unwrap_or(0.0))
185
+ } else {
186
+ Value::Unknown
187
+ }
188
+ } else {
189
+ Value::Unknown
190
+ }
191
+ }
192
+ TokenKind::Number => {
193
+ // Possibly a float with dot
194
+ let mut number_str = vtok.lexeme.clone();
195
+ local.advance();
196
+ if let Some(dot) = local.peek_clone() {
197
+ if dot.kind == TokenKind::Dot {
198
+ local.advance();
199
+ if let Some(frac) = local.peek_clone() {
200
+ if frac.kind == TokenKind::Number {
201
+ number_str.push('.');
202
+ number_str.push_str(&frac.lexeme);
203
+ local.advance();
204
+ }
205
+ }
206
+ }
207
+ }
208
+ Value::Number(number_str.parse::<f32>().unwrap_or(0.0))
209
+ }
210
+ TokenKind::Identifier => {
211
+ local.advance();
212
+ Value::Identifier(vtok.lexeme.clone())
213
+ }
214
+ TokenKind::String => {
215
+ local.advance();
216
+ Value::String(vtok.lexeme.clone())
217
+ }
218
+ _ => {
219
+ local.advance();
220
+ Value::Unknown
221
+ }
222
+ }
223
+ } else {
224
+ Value::Null
225
+ };
226
+
227
+ envelope.insert(percent_key, value);
228
+ }
229
+
230
+ params.insert(name_tok.lexeme.clone(), Value::Map(envelope));
231
+ }
232
+ _ => {
233
+ local.advance();
234
+ }
235
+ }
236
+ }
237
+
238
+ let mut value_map = HashMap::new();
239
+ value_map.insert(
240
+ "target".to_string(),
241
+ Value::String(target_token.lexeme.clone()),
242
+ );
243
+ value_map.insert("params".to_string(), Value::Map(params));
244
+
245
+ Statement {
246
+ kind: StatementKind::Automate {
247
+ target: target_token.lexeme.clone(),
248
+ },
249
+ value: Value::Map(value_map),
250
+ indent: current_token.indent,
251
+ line: current_token.line,
252
+ column: current_token.column,
253
+ }
254
+ }
@@ -1,41 +1,91 @@
1
- use crate::core::{
2
- lexer::token::{ Token, TokenKind },
3
- parser::{ statement::{ Statement, StatementKind }, driver::Parser },
4
- shared::value::Value,
5
- store::global::GlobalStore,
6
- };
7
-
8
- pub fn parse_call_token(
9
- parser: &mut Parser,
10
- current_token: Token,
11
- global_store: &mut GlobalStore
12
- ) -> Statement {
13
- parser.advance(); // consume "call"
14
-
15
- let value = if let Some(token) = parser.peek_clone() {
16
- parser.advance();
17
- match token.kind {
18
- TokenKind::Identifier => Value::Identifier(token.lexeme.clone()),
19
- TokenKind::String => Value::String(token.lexeme.clone()),
20
- _ => {
21
- return Statement::error(
22
- token,
23
- "Expected identifier or string after 'call'".to_string()
24
- );
25
- }
26
- }
27
- } else {
28
- return Statement::error(
29
- current_token,
30
- "Expected identifier or string after 'call'".to_string()
31
- );
32
- };
33
-
34
- return Statement {
35
- kind: StatementKind::Call,
36
- value,
37
- indent: current_token.indent,
38
- line: current_token.line,
39
- column: current_token.column,
40
- };
41
- }
1
+ use crate::core::{
2
+ lexer::token::{Token, TokenKind},
3
+ parser::{
4
+ driver::Parser,
5
+ statement::{Statement, StatementKind},
6
+ },
7
+ store::global::GlobalStore,
8
+ };
9
+ use devalang_types::Value;
10
+
11
+ pub fn parse_call_token(
12
+ parser: &mut Parser,
13
+ current_token: Token,
14
+ _global_store: &mut GlobalStore,
15
+ ) -> Statement {
16
+ parser.advance(); // consume "call"
17
+
18
+ // Expect function name
19
+ let name_token = match parser.peek_clone() {
20
+ Some(t) => t,
21
+ None => {
22
+ return crate::core::parser::statement::error_from_token(
23
+ current_token,
24
+ "Expected function name after 'call'".to_string(),
25
+ );
26
+ }
27
+ };
28
+
29
+ if name_token.kind != TokenKind::Identifier {
30
+ return crate::core::parser::statement::error_from_token(
31
+ name_token,
32
+ "Expected function name to be an identifier".to_string(),
33
+ );
34
+ }
35
+
36
+ let func_name = name_token.lexeme.clone();
37
+ parser.advance(); // consume function name
38
+
39
+ // Expect '('
40
+ let mut args: Vec<Value> = Vec::new();
41
+ if let Some(open_paren) = parser.peek_clone() {
42
+ if open_paren.kind == TokenKind::LParen {
43
+ parser.advance(); // consume '('
44
+
45
+ // Collect args until ')'
46
+ while let Some(token) = parser.peek_clone() {
47
+ if token.kind == TokenKind::RParen {
48
+ parser.advance(); // consume ')'
49
+ break;
50
+ }
51
+
52
+ match token.kind {
53
+ TokenKind::Number => {
54
+ if let Ok(num) = token.lexeme.parse::<f32>() {
55
+ args.push(Value::Number(num));
56
+ }
57
+ parser.advance();
58
+ }
59
+ TokenKind::String => {
60
+ args.push(Value::String(token.lexeme.clone()));
61
+ parser.advance();
62
+ }
63
+ TokenKind::Identifier => {
64
+ args.push(Value::Identifier(token.lexeme.clone()));
65
+ parser.advance();
66
+ }
67
+ TokenKind::Comma => {
68
+ parser.advance(); // skip comma
69
+ }
70
+ _ => {
71
+ return crate::core::parser::statement::error_from_token(
72
+ token,
73
+ "Unexpected token in call arguments".to_string(),
74
+ );
75
+ }
76
+ }
77
+ }
78
+ }
79
+ }
80
+
81
+ Statement {
82
+ kind: StatementKind::Call {
83
+ name: func_name,
84
+ args,
85
+ },
86
+ value: Value::Null,
87
+ indent: current_token.indent,
88
+ line: current_token.line,
89
+ column: current_token.column,
90
+ }
91
+ }
@@ -0,0 +1,70 @@
1
+ use crate::core::{
2
+ lexer::token::TokenKind,
3
+ parser::{
4
+ driver::Parser,
5
+ statement::{Statement, StatementKind},
6
+ },
7
+ store::global::GlobalStore,
8
+ };
9
+ use devalang_types::Value;
10
+
11
+ pub fn parse_emit_token(
12
+ parser: &mut Parser,
13
+ current: crate::core::lexer::token::Token,
14
+ _global_store: &mut GlobalStore,
15
+ ) -> Statement {
16
+ parser.advance(); // consume 'emit'
17
+
18
+ let Some(ev) = parser.peek_clone() else {
19
+ return crate::core::parser::statement::error_from_token(
20
+ current,
21
+ "Expected event name after 'emit'".into(),
22
+ );
23
+ };
24
+ if ev.kind != TokenKind::Identifier {
25
+ return crate::core::parser::statement::error_from_token(
26
+ ev.clone(),
27
+ "Expected identifier as event name".into(),
28
+ );
29
+ }
30
+ let event_name = ev.lexeme.clone();
31
+ parser.advance(); // consume event name
32
+
33
+ // Optional payload on same line: number|string|identifier|map|array
34
+ let mut payload: Option<Value> = None;
35
+ if let Some(tok) = parser.peek_clone() {
36
+ if tok.line == ev.line {
37
+ let val = match tok.kind {
38
+ TokenKind::String => {
39
+ parser.advance();
40
+ Value::String(tok.lexeme.clone())
41
+ }
42
+ TokenKind::Number => {
43
+ parser.advance();
44
+ Value::Number(tok.lexeme.parse().unwrap_or(0.0))
45
+ }
46
+ TokenKind::Identifier => {
47
+ parser.advance();
48
+ Value::Identifier(tok.lexeme.clone())
49
+ }
50
+ TokenKind::LBrace => parser.parse_map_value().unwrap_or(Value::Null),
51
+ TokenKind::LBracket => parser.parse_array_value().unwrap_or(Value::Null),
52
+ _ => Value::Null,
53
+ };
54
+ if val != Value::Null {
55
+ payload = Some(val);
56
+ }
57
+ }
58
+ }
59
+
60
+ Statement {
61
+ kind: StatementKind::Emit {
62
+ event: event_name,
63
+ payload: payload.clone(),
64
+ },
65
+ value: payload.unwrap_or(Value::Null),
66
+ indent: current.indent,
67
+ line: current.line,
68
+ column: current.column,
69
+ }
70
+ }
@@ -0,0 +1,113 @@
1
+ use devalang_types::Value;
2
+
3
+ use crate::core::{
4
+ lexer::token::TokenKind,
5
+ parser::{
6
+ driver::Parser,
7
+ statement::{Statement, StatementKind},
8
+ },
9
+ store::global::GlobalStore,
10
+ };
11
+
12
+ pub fn parse_function_token(parser: &mut Parser, global_store: &mut GlobalStore) -> Statement {
13
+ parser.advance(); // consume 'fn'
14
+
15
+ let fn_token = match parser.previous_clone() {
16
+ Some(tok) => tok,
17
+ None => return Statement::unknown(),
18
+ };
19
+
20
+ let name_token = match parser.peek_clone() {
21
+ Some(tok) => tok,
22
+ None => {
23
+ return crate::core::parser::statement::error_from_token(
24
+ fn_token,
25
+ "Expected function name after 'fn'".to_string(),
26
+ );
27
+ }
28
+ };
29
+
30
+ if name_token.kind != TokenKind::Identifier {
31
+ return crate::core::parser::statement::error_from_token(
32
+ name_token.clone(),
33
+ "Expected function name to be an identifier".to_string(),
34
+ );
35
+ }
36
+
37
+ let function_name = name_token.lexeme.clone();
38
+ parser.advance(); // consume function name
39
+
40
+ let mut parameters = Vec::new();
41
+
42
+ // Expect '('
43
+ if parser.peek_kind() != Some(TokenKind::LParen) {
44
+ return crate::core::parser::statement::error_from_token(
45
+ name_token.clone(),
46
+ "Expected '(' after function name".to_string(),
47
+ );
48
+ }
49
+ parser.advance(); // consume '('
50
+
51
+ // Parse parameters until ')'
52
+ let tokens = parser.collect_until(|t| t.kind == TokenKind::RParen || t.kind == TokenKind::EOF);
53
+ for token in tokens {
54
+ if token.kind == TokenKind::Identifier {
55
+ parameters.push(token.lexeme.clone());
56
+ }
57
+ }
58
+
59
+ if parser.peek_kind() == Some(TokenKind::RParen) {
60
+ parser.advance(); // consume ')'
61
+ } else {
62
+ return crate::core::parser::statement::error_from_token(
63
+ name_token.clone(),
64
+ "Expected ')' after parameters".to_string(),
65
+ );
66
+ }
67
+
68
+ // Expect colon
69
+ if parser.peek_kind() != Some(TokenKind::Colon) {
70
+ return crate::core::parser::statement::error_from_token(
71
+ name_token.clone(),
72
+ "Expected ':' after ')'".to_string(),
73
+ );
74
+ }
75
+ parser.advance(); // consume ':'
76
+
77
+ // Collect ALL tokens indented after this line until Dedent
78
+ let base_indent = fn_token.indent;
79
+ let mut body_tokens = Vec::new();
80
+
81
+ while let Some(tok) = parser.peek() {
82
+ if tok.kind == TokenKind::Dedent && tok.indent <= base_indent {
83
+ break;
84
+ }
85
+ if let Some(t) = parser.advance() {
86
+ body_tokens.push(t.clone());
87
+ } else {
88
+ break;
89
+ }
90
+ }
91
+
92
+ // Parse those tokens into block statements
93
+ let body = parser.parse_block(body_tokens.clone(), global_store);
94
+
95
+ // Skip Dedent if present
96
+ if let Some(tok) = parser.peek() {
97
+ if tok.kind == TokenKind::Dedent {
98
+ parser.advance();
99
+ }
100
+ }
101
+
102
+ Statement {
103
+ kind: StatementKind::Function {
104
+ name: function_name.clone(),
105
+ parameters: parameters.clone(),
106
+ body: body.clone(),
107
+ },
108
+ value: Value::Null,
109
+ indent: fn_token.indent,
110
+ line: fn_token.line,
111
+ column: fn_token.column,
112
+ }
113
+ }