@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
@@ -1,75 +1,89 @@
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
- use std::collections::HashMap;
8
-
9
- pub fn parse_group_token(
10
- parser: &mut Parser,
11
- current_token: Token,
12
- global_store: &mut GlobalStore
13
- ) -> Statement {
14
- parser.advance(); // consume "group"
15
-
16
- let Some(identifier_token) = parser.peek_clone() else {
17
- return Statement::error(current_token, "Expected identifier after 'group'".to_string());
18
- };
19
-
20
- if identifier_token.kind != TokenKind::Identifier && identifier_token.kind != TokenKind::String {
21
- return Statement::error(identifier_token, "Expected valid identifier".to_string());
22
- }
23
-
24
- parser.advance(); // consume identifier
25
-
26
- let Some(colon_token) = parser.peek_clone() else {
27
- return Statement::error(
28
- identifier_token,
29
- "Expected ':' after group identifier".to_string()
30
- );
31
- };
32
-
33
- if colon_token.kind != TokenKind::Colon {
34
- return Statement::error(
35
- colon_token.clone(),
36
- "Expected ':' after group identifier".to_string()
37
- );
38
- }
39
-
40
- parser.advance(); // consume ':'
41
-
42
- let base_indent = current_token.indent;
43
-
44
- // Clone without consuming tokens
45
- let mut index = parser.token_index;
46
- let mut tokens_inside_group = Vec::new();
47
-
48
- while index < parser.tokens.len() {
49
- let token = parser.tokens[index].clone();
50
-
51
- if token.indent <= base_indent && token.kind != TokenKind::Newline {
52
- break;
53
- }
54
-
55
- tokens_inside_group.push(token);
56
- index += 1;
57
- }
58
-
59
- // Advance index once to skip the processed tokens
60
- parser.token_index = index;
61
-
62
- let body = parser.parse_block(tokens_inside_group, global_store);
63
-
64
- let mut value_map = HashMap::new();
65
- value_map.insert("identifier".to_string(), Value::String(identifier_token.lexeme.clone()));
66
- value_map.insert("body".to_string(), Value::Block(body));
67
-
68
- return Statement {
69
- kind: StatementKind::Group,
70
- value: Value::Map(value_map),
71
- indent: current_token.indent,
72
- line: current_token.line,
73
- column: current_token.column,
74
- };
75
- }
1
+ use devalang_types::Value;
2
+
3
+ use crate::core::{
4
+ lexer::token::{Token, TokenKind},
5
+ parser::{
6
+ driver::Parser,
7
+ statement::{Statement, StatementKind},
8
+ },
9
+ store::global::GlobalStore,
10
+ };
11
+ use std::collections::HashMap;
12
+
13
+ pub fn parse_group_token(
14
+ parser: &mut Parser,
15
+ current_token: Token,
16
+ global_store: &mut GlobalStore,
17
+ ) -> Statement {
18
+ parser.advance(); // consume "group"
19
+
20
+ let Some(identifier_token) = parser.peek_clone() else {
21
+ return crate::core::parser::statement::error_from_token(
22
+ current_token,
23
+ "Expected identifier after 'group'".to_string(),
24
+ );
25
+ };
26
+
27
+ if identifier_token.kind != TokenKind::Identifier && identifier_token.kind != TokenKind::String
28
+ {
29
+ return crate::core::parser::statement::error_from_token(
30
+ identifier_token,
31
+ "Expected valid identifier".to_string(),
32
+ );
33
+ }
34
+
35
+ parser.advance(); // consume identifier
36
+
37
+ let Some(colon_token) = parser.peek_clone() else {
38
+ return crate::core::parser::statement::error_from_token(
39
+ identifier_token,
40
+ "Expected ':' after group identifier".to_string(),
41
+ );
42
+ };
43
+
44
+ if colon_token.kind != TokenKind::Colon {
45
+ return crate::core::parser::statement::error_from_token(
46
+ colon_token.clone(),
47
+ "Expected ':' after group identifier".to_string(),
48
+ );
49
+ }
50
+
51
+ parser.advance(); // consume ':'
52
+
53
+ let base_indent = current_token.indent;
54
+
55
+ // Clone without consuming tokens
56
+ let mut index = parser.token_index;
57
+ let mut tokens_inside_group = Vec::new();
58
+
59
+ while index < parser.tokens.len() {
60
+ let token = parser.tokens[index].clone();
61
+
62
+ if token.indent <= base_indent && token.kind != TokenKind::Newline {
63
+ break;
64
+ }
65
+
66
+ tokens_inside_group.push(token);
67
+ index += 1;
68
+ }
69
+
70
+ // Advance index once to skip the processed tokens
71
+ parser.token_index = index;
72
+
73
+ let body = parser.parse_block(tokens_inside_group, global_store);
74
+
75
+ let mut value_map = HashMap::new();
76
+ value_map.insert(
77
+ "identifier".to_string(),
78
+ Value::String(identifier_token.lexeme.clone()),
79
+ );
80
+ value_map.insert("body".to_string(), Value::Block(body));
81
+
82
+ Statement {
83
+ kind: StatementKind::Group,
84
+ value: Value::Map(value_map),
85
+ indent: current_token.indent,
86
+ line: current_token.line,
87
+ column: current_token.column,
88
+ }
89
+ }
@@ -1,133 +1,173 @@
1
- use std::collections::HashMap;
2
-
3
- use crate::core::{
4
- lexer::token::{ Token, TokenKind },
5
- parser::{
6
- driver::Parser,
7
- handler::identifier::synth::parse_synth_token,
8
- statement::{ Statement, StatementKind },
9
- },
10
- shared::value::Value,
11
- store::global::GlobalStore,
12
- };
13
-
14
- pub fn parse_let_token(
15
- parser: &mut Parser,
16
- current_token: Token,
17
- global_store: &mut GlobalStore
18
- ) -> Statement {
19
- parser.advance(); // consume "let"
20
-
21
- let identifier = if let Some(token) = parser.peek_clone() {
22
- if token.kind == TokenKind::Identifier {
23
- parser.advance();
24
- token.lexeme.clone()
25
- } else {
26
- return Statement::error(token, "Expected identifier after 'let'".to_string());
27
- }
28
- } else {
29
- return Statement::error(current_token, "Expected identifier after 'let'".to_string());
30
- };
31
-
32
- if !parser.match_token(TokenKind::Equals) {
33
- return Statement::error(current_token, "Expected '=' after identifier".to_string());
34
- }
35
-
36
- if let Some(token) = parser.peek_clone() {
37
- if token.kind == TokenKind::Synth {
38
- let synth_stmt = parse_synth_token(parser, token.clone(), global_store);
39
-
40
- return Statement {
41
- kind: StatementKind::Let { name: identifier },
42
- value: synth_stmt.value,
43
- indent: current_token.indent,
44
- line: current_token.line,
45
- column: current_token.column,
46
- };
47
- }
48
- }
49
-
50
- let value = match parser.peek_clone() {
51
- Some(token) if token.kind == TokenKind::Identifier => {
52
- parser.advance();
53
- Value::Identifier(token.lexeme.clone())
54
- }
55
- Some(token) if token.kind == TokenKind::String => {
56
- parser.advance();
57
- Value::String(token.lexeme.clone())
58
- }
59
- Some(token) if token.kind == TokenKind::Number => {
60
- parser.advance();
61
- Value::Number(token.lexeme.parse().unwrap_or(0.0))
62
- }
63
- Some(token) if token.kind == TokenKind::Boolean => {
64
- parser.advance();
65
- Value::Boolean(token.lexeme.parse().unwrap_or(false))
66
- }
67
- Some(token) if token.kind == TokenKind::LBrace => {
68
- parser.advance(); // consume '{'
69
- let mut map = HashMap::new();
70
-
71
- while let Some(key_token) = parser.peek_clone() {
72
- if key_token.kind == TokenKind::RBrace {
73
- parser.advance(); // consume '}'
74
- break;
75
- }
76
-
77
- if key_token.kind != TokenKind::Identifier {
78
- return Statement::error(token, "Expected key identifier in map".to_string());
79
- }
80
- parser.advance();
81
- let key = key_token.lexeme.clone();
82
-
83
- if !parser.match_token(TokenKind::Colon) {
84
- let message = format!("Expected ':' after key '{}'", key);
85
- return Statement::error(token, message);
86
- }
87
-
88
- let val = match parser.peek_clone() {
89
- Some(t) if t.kind == TokenKind::Number => {
90
- parser.advance();
91
- Value::Number(t.lexeme.parse().unwrap_or(0.0))
92
- }
93
- Some(t) if t.kind == TokenKind::String => {
94
- parser.advance();
95
- Value::String(t.lexeme.clone())
96
- }
97
- Some(t) if t.kind == TokenKind::Identifier => {
98
- parser.advance();
99
- Value::Identifier(t.lexeme.clone())
100
- }
101
- _ => Value::Null,
102
- };
103
-
104
- if val == Value::Null {
105
- let message = format!("Invalid value for key '{}'", key);
106
- return Statement::error(token, message);
107
- }
108
-
109
- map.insert(key, val);
110
-
111
- if let Some(t) = parser.peek() {
112
- if t.kind == TokenKind::Comma {
113
- parser.advance(); // skip comma
114
- }
115
- }
116
- }
117
-
118
- Value::Map(map)
119
- }
120
- other => {
121
- let message = format!("Unexpected value token in let: {:?}", other);
122
- return Statement::error(current_token, message);
123
- }
124
- };
125
-
126
- Statement {
127
- kind: StatementKind::Let { name: identifier },
128
- value,
129
- indent: current_token.indent,
130
- line: current_token.line,
131
- column: current_token.column,
132
- }
133
- }
1
+ use std::collections::HashMap;
2
+
3
+ use crate::core::{
4
+ lexer::token::{Token, TokenKind},
5
+ parser::{
6
+ driver::Parser,
7
+ handler::{dot::parse_dot_token, identifier::synth::parse_synth_token},
8
+ statement::{Statement, StatementKind},
9
+ },
10
+ store::global::GlobalStore,
11
+ };
12
+ use devalang_types::Value;
13
+
14
+ pub fn parse_let_token(
15
+ parser: &mut Parser,
16
+ current_token: Token,
17
+ global_store: &mut GlobalStore,
18
+ ) -> Statement {
19
+ parser.advance(); // consume "let"
20
+
21
+ let identifier = if let Some(token) = parser.peek_clone() {
22
+ if token.kind == TokenKind::Identifier {
23
+ parser.advance();
24
+ token.lexeme.clone()
25
+ } else {
26
+ return crate::core::parser::statement::error_from_token(
27
+ token,
28
+ "Expected identifier after 'let'".to_string(),
29
+ );
30
+ }
31
+ } else {
32
+ return crate::core::parser::statement::error_from_token(
33
+ current_token,
34
+ "Expected identifier after 'let'".to_string(),
35
+ );
36
+ };
37
+
38
+ if !parser.match_token(TokenKind::Equals) {
39
+ return crate::core::parser::statement::error_from_token(
40
+ current_token,
41
+ "Expected '=' after identifier".to_string(),
42
+ );
43
+ }
44
+
45
+ // If RHS begins with '$' or contains expression tokens ('+', '-', '*', '/', '(', '['),
46
+ // collect the rest of the line as a raw expression string.
47
+ if let Some(tok) = parser.peek_clone() {
48
+ let line = tok.line;
49
+ if tok.lexeme.starts_with('$')
50
+ || matches!(
51
+ tok.kind,
52
+ TokenKind::Identifier | TokenKind::Number | TokenKind::LParen | TokenKind::LBracket
53
+ )
54
+ {
55
+ // Collect tokens until end of the current line
56
+ let collected = parser.collect_until(|t| {
57
+ t.line != line || matches!(t.kind, TokenKind::Newline | TokenKind::EOF)
58
+ });
59
+ let mut text = String::new();
60
+ for t in collected.iter() {
61
+ if matches!(t.kind, TokenKind::Newline | TokenKind::EOF) {
62
+ break;
63
+ }
64
+ text.push_str(&t.lexeme);
65
+ }
66
+ return Statement {
67
+ kind: StatementKind::Let { name: identifier },
68
+ value: Value::String(text.trim().to_string()),
69
+ indent: current_token.indent,
70
+ line: current_token.line,
71
+ column: current_token.column,
72
+ };
73
+ }
74
+ }
75
+
76
+ let value = match parser.peek_clone() {
77
+ Some(token) if token.kind == TokenKind::Dot => {
78
+ let dot_stmt = parse_dot_token(parser, global_store);
79
+ Value::Statement(Box::new(dot_stmt))
80
+ }
81
+ Some(token) if token.kind == TokenKind::Synth => {
82
+ let synth_stmt = parse_synth_token(parser, token.clone(), global_store);
83
+ Value::Statement(Box::new(synth_stmt))
84
+ }
85
+ Some(token) if token.kind == TokenKind::Identifier => {
86
+ parser.advance();
87
+ Value::Identifier(token.lexeme.clone())
88
+ }
89
+ Some(token) if token.kind == TokenKind::String => {
90
+ parser.advance();
91
+ Value::String(token.lexeme.clone())
92
+ }
93
+ Some(token) if token.kind == TokenKind::Number => {
94
+ parser.advance();
95
+ Value::Number(token.lexeme.parse().unwrap_or(0.0))
96
+ }
97
+ Some(token) if token.kind == TokenKind::Boolean => {
98
+ parser.advance();
99
+ Value::Boolean(token.lexeme.parse().unwrap_or(false))
100
+ }
101
+ Some(token) if token.kind == TokenKind::LBrace => {
102
+ parser.advance();
103
+
104
+ let mut map = HashMap::new();
105
+
106
+ while let Some(key_token) = parser.peek_clone() {
107
+ if key_token.kind == TokenKind::RBrace {
108
+ parser.advance(); // consume '}'
109
+ break;
110
+ }
111
+
112
+ if key_token.kind != TokenKind::Identifier {
113
+ return crate::core::parser::statement::error_from_token(
114
+ token,
115
+ "Expected key identifier in map".to_string(),
116
+ );
117
+ }
118
+ parser.advance();
119
+ let key = key_token.lexeme.clone();
120
+
121
+ if !parser.match_token(TokenKind::Colon) {
122
+ let message = format!("Expected ':' after key '{}'", key);
123
+ return crate::core::parser::statement::error_from_token(token, message);
124
+ }
125
+
126
+ let val = match parser.peek_clone() {
127
+ Some(t) if t.kind == TokenKind::Number => {
128
+ parser.advance();
129
+ Value::Number(t.lexeme.parse().unwrap_or(0.0))
130
+ }
131
+ Some(t) if t.kind == TokenKind::String => {
132
+ parser.advance();
133
+ Value::String(t.lexeme.clone())
134
+ }
135
+ Some(t) if t.kind == TokenKind::Identifier => {
136
+ parser.advance();
137
+ Value::Identifier(t.lexeme.clone())
138
+ }
139
+ _ => Value::Null,
140
+ };
141
+
142
+ if val == Value::Null {
143
+ let message = format!("Invalid value for key '{}'", key);
144
+ return crate::core::parser::statement::error_from_token(token, message);
145
+ }
146
+
147
+ map.insert(key, val);
148
+
149
+ if let Some(t) = parser.peek() {
150
+ if t.kind == TokenKind::Comma {
151
+ parser.advance(); // skip comma
152
+ }
153
+ }
154
+ }
155
+
156
+ Value::Map(map)
157
+ }
158
+ _ => {
159
+ return crate::core::parser::statement::error_from_token(
160
+ current_token,
161
+ "Unhandled value type after '='".to_string(),
162
+ );
163
+ }
164
+ };
165
+
166
+ Statement {
167
+ kind: StatementKind::Let { name: identifier },
168
+ value,
169
+ indent: current_token.indent,
170
+ line: current_token.line,
171
+ column: current_token.column,
172
+ }
173
+ }
@@ -1,51 +1,55 @@
1
- pub mod let_;
2
- pub mod group;
3
- pub mod call;
4
- pub mod spawn;
5
- pub mod sleep;
6
- pub mod synth;
7
-
8
- use crate::core::{
9
- parser::{
10
- driver::Parser,
11
- handler::{
12
- identifier::{
13
- call::parse_call_token,
14
- group::parse_group_token,
15
- let_::parse_let_token,
16
- sleep::parse_sleep_token,
17
- spawn::parse_spawn_token,
18
- synth::parse_synth_token
19
- },
20
- },
21
- statement::Statement,
22
- },
23
- store::global::GlobalStore,
24
- };
25
-
26
- pub fn parse_identifier_token(parser: &mut Parser, global_store: &mut GlobalStore) -> Statement {
27
- let Some(current_token) = parser.peek_clone() else {
28
- return Statement::unknown();
29
- };
30
-
31
- let current_token_clone = current_token.clone();
32
- let current_token_lexeme = current_token_clone.lexeme.clone();
33
-
34
- let statement = match current_token_lexeme.as_str() {
35
- "let" => parse_let_token(parser, current_token_clone, global_store),
36
- "group" => parse_group_token(parser, current_token_clone, global_store),
37
- "call" => parse_call_token(parser, current_token_clone, global_store),
38
- "spawn" => parse_spawn_token(parser, current_token_clone, global_store),
39
- "sleep" => parse_sleep_token(parser, current_token_clone, global_store),
40
- "synth" => parse_synth_token(parser, current_token_clone, global_store),
41
- _ => {
42
- parser.advance(); // consume identifier
43
-
44
- println!("Unrecognized identifier: {}", current_token_lexeme);
45
-
46
- return Statement::error(current_token_clone, "Unexpected identifier".to_string());
47
- }
48
- };
49
-
50
- return statement;
51
- }
1
+ pub mod automate;
2
+ pub mod call;
3
+ pub mod emit;
4
+ pub mod function;
5
+ pub mod group;
6
+ pub mod let_;
7
+ pub mod on;
8
+ pub mod print;
9
+ pub mod sleep;
10
+ pub mod spawn;
11
+ pub mod synth;
12
+
13
+ use crate::core::{
14
+ parser::{
15
+ driver::Parser,
16
+ handler::identifier::{
17
+ automate::parse_automate_token, call::parse_call_token, emit::parse_emit_token,
18
+ group::parse_group_token, let_::parse_let_token, on::parse_on_token,
19
+ print::parse_print_token, sleep::parse_sleep_token, spawn::parse_spawn_token,
20
+ synth::parse_synth_token,
21
+ },
22
+ statement::Statement,
23
+ },
24
+ store::global::GlobalStore,
25
+ };
26
+
27
+ pub fn parse_identifier_token(parser: &mut Parser, global_store: &mut GlobalStore) -> Statement {
28
+ let Some(current_token) = parser.peek_clone() else {
29
+ return Statement::unknown();
30
+ };
31
+
32
+ let current_token_clone = current_token.clone();
33
+ let current_token_lexeme = current_token_clone.lexeme.clone();
34
+
35
+ match current_token_lexeme.as_str() {
36
+ "let" => parse_let_token(parser, current_token_clone, global_store),
37
+ "group" => parse_group_token(parser, current_token_clone, global_store),
38
+ "call" => parse_call_token(parser, current_token_clone, global_store),
39
+ "spawn" => parse_spawn_token(parser, current_token_clone, global_store),
40
+ "sleep" => parse_sleep_token(parser, current_token_clone, global_store),
41
+ "synth" => parse_synth_token(parser, current_token_clone, global_store),
42
+ "automate" => parse_automate_token(parser, current_token_clone, global_store),
43
+ "print" => parse_print_token(parser, current_token_clone, global_store),
44
+ "on" => parse_on_token(parser, global_store),
45
+ "emit" => parse_emit_token(parser, current_token_clone, global_store),
46
+ _ => {
47
+ parser.advance(); // consume identifier
48
+
49
+ crate::core::parser::statement::error_from_token(
50
+ current_token_clone,
51
+ "Unexpected identifier".to_string(),
52
+ )
53
+ }
54
+ }
55
+ }