@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,107 @@
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_on_token(parser: &mut Parser, _global_store: &mut GlobalStore) -> Statement {
13
+ // consume 'on'
14
+ let on_tok = match parser.peek_clone() {
15
+ Some(tok) => tok,
16
+ None => return Statement::unknown(),
17
+ };
18
+ parser.advance();
19
+
20
+ // Expect event name identifier
21
+ let event_tok = match parser.peek_clone() {
22
+ Some(tok) if tok.kind == TokenKind::Identifier => tok,
23
+ Some(other) => {
24
+ return crate::core::parser::statement::error_from_token(
25
+ other,
26
+ "Expected event name after 'on'".to_string(),
27
+ );
28
+ }
29
+ None => {
30
+ return crate::core::parser::statement::error_from_token(
31
+ on_tok,
32
+ "Expected event name after 'on'".to_string(),
33
+ );
34
+ }
35
+ };
36
+ let event_name = event_tok.lexeme.clone();
37
+ parser.advance();
38
+
39
+ // Optional parenthesized args on same line
40
+ let mut args: Option<Vec<Value>> = None;
41
+ if parser.peek_kind() == Some(TokenKind::LParen) {
42
+ parser.advance(); // '('
43
+ let mut collected: Vec<Value> = Vec::new();
44
+ // Collect tokens until ')', supporting numbers and identifiers separated by comma
45
+ while let Some(tok) = parser.peek_clone() {
46
+ match tok.kind {
47
+ TokenKind::RParen => {
48
+ parser.advance();
49
+ break;
50
+ }
51
+ TokenKind::Number => {
52
+ parser.advance();
53
+ collected.push(Value::Number(tok.lexeme.parse().unwrap_or(0.0)));
54
+ }
55
+ TokenKind::Identifier => {
56
+ parser.advance();
57
+ collected.push(Value::Identifier(tok.lexeme));
58
+ }
59
+ TokenKind::Comma => {
60
+ parser.advance();
61
+ }
62
+ TokenKind::Whitespace | TokenKind::Newline => {
63
+ parser.advance();
64
+ }
65
+ _ => {
66
+ break;
67
+ }
68
+ }
69
+ }
70
+ if !collected.is_empty() {
71
+ args = Some(collected);
72
+ }
73
+ }
74
+
75
+ // Expect ':' then block
76
+ if parser.peek_kind() != Some(TokenKind::Colon) {
77
+ return crate::core::parser::statement::error_from_token(
78
+ event_tok,
79
+ "Expected ':' after event name".to_string(),
80
+ );
81
+ }
82
+ parser.advance(); // consume ':'
83
+
84
+ let base_indent = on_tok.indent;
85
+ let block_tokens = parser.collect_block_tokens(base_indent);
86
+ // Parse body within current store context
87
+ let body = parser.parse_block(block_tokens, _global_store);
88
+
89
+ let stmt = Statement {
90
+ kind: StatementKind::On {
91
+ event: event_name,
92
+ args,
93
+ body,
94
+ },
95
+ value: Value::Null,
96
+ indent: on_tok.indent,
97
+ line: on_tok.line,
98
+ column: on_tok.column,
99
+ };
100
+
101
+ // Register in global store for later emission
102
+ if let StatementKind::On { event, .. } = &stmt.kind {
103
+ _global_store.register_event_handler(event, stmt.clone());
104
+ }
105
+
106
+ stmt
107
+ }
@@ -0,0 +1,49 @@
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_print_token(
12
+ parser: &mut Parser,
13
+ current_token: Token,
14
+ _global_store: &mut GlobalStore,
15
+ ) -> Statement {
16
+ // consume 'print'
17
+ parser.advance();
18
+
19
+ let collected = parser.collect_until(|t| matches!(t.kind, TokenKind::Newline | TokenKind::EOF));
20
+ // Accept: print <identifier|string|number|expression>
21
+ let value = if collected.len() == 1 {
22
+ match collected[0].kind {
23
+ TokenKind::Identifier => Value::Identifier(collected[0].lexeme.clone()),
24
+ TokenKind::String => Value::String(collected[0].lexeme.clone()),
25
+ TokenKind::Number => {
26
+ let n = collected[0].lexeme.parse::<f32>().unwrap_or(0.0);
27
+ Value::Number(n)
28
+ }
29
+ _ => Value::String(collected[0].lexeme.clone()),
30
+ }
31
+ } else {
32
+ // Join tokens with spaces to preserve readability for expressions/text
33
+ let text = collected
34
+ .iter()
35
+ .filter(|t| !matches!(t.kind, TokenKind::Newline | TokenKind::EOF))
36
+ .map(|t| t.lexeme.clone())
37
+ .collect::<Vec<_>>()
38
+ .join(" ");
39
+ Value::String(text.trim().to_string())
40
+ };
41
+
42
+ Statement {
43
+ kind: StatementKind::Print,
44
+ value,
45
+ indent: current_token.indent,
46
+ line: current_token.line,
47
+ column: current_token.column,
48
+ }
49
+ }
@@ -1,33 +1,43 @@
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_sleep_token(
9
- parser: &mut Parser,
10
- current_token: Token,
11
- global_store: &mut GlobalStore
12
- ) -> Statement {
13
- parser.advance(); // consume "sleep"
14
-
15
- let duration = if let Some(token) = parser.peek_clone() {
16
- if token.kind == TokenKind::Number {
17
- parser.advance();
18
- token.lexeme.parse().unwrap_or(0.0)
19
- } else {
20
- return Statement::error(token, "Expected number after 'sleep'".to_string());
21
- }
22
- } else {
23
- return Statement::error(current_token, "Expected number after 'sleep'".to_string());
24
- };
25
-
26
- return Statement {
27
- kind: StatementKind::Sleep,
28
- value: Value::Number(duration),
29
- indent: current_token.indent,
30
- line: current_token.line,
31
- column: current_token.column,
32
- };
33
- }
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
+
12
+ pub fn parse_sleep_token(
13
+ parser: &mut Parser,
14
+ current_token: Token,
15
+ _global_store: &mut GlobalStore,
16
+ ) -> Statement {
17
+ parser.advance(); // consume "sleep"
18
+
19
+ let duration = if let Some(token) = parser.peek_clone() {
20
+ if token.kind == TokenKind::Number {
21
+ parser.advance();
22
+ token.lexeme.parse().unwrap_or(0.0)
23
+ } else {
24
+ return crate::core::parser::statement::error_from_token(
25
+ token,
26
+ "Expected number after 'sleep'".to_string(),
27
+ );
28
+ }
29
+ } else {
30
+ return crate::core::parser::statement::error_from_token(
31
+ current_token,
32
+ "Expected number after 'sleep'".to_string(),
33
+ );
34
+ };
35
+
36
+ Statement {
37
+ kind: StatementKind::Sleep,
38
+ value: Value::Number(duration),
39
+ indent: current_token.indent,
40
+ line: current_token.line,
41
+ column: current_token.column,
42
+ }
43
+ }
@@ -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_spawn_token(
9
- parser: &mut Parser,
10
- current_token: Token,
11
- global_store: &mut GlobalStore
12
- ) -> Statement {
13
- parser.advance(); // consume "spawn"
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 'spawn'".to_string()
24
- );
25
- }
26
- }
27
- } else {
28
- return Statement::error(
29
- current_token,
30
- "Expected identifier or string after 'spawn'".to_string()
31
- );
32
- };
33
-
34
- return Statement {
35
- kind: StatementKind::Spawn,
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_spawn_token(
12
+ parser: &mut Parser,
13
+ current_token: Token,
14
+ _global_store: &mut GlobalStore,
15
+ ) -> Statement {
16
+ parser.advance(); // consume "spawn"
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 'spawn'".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 spawn arguments".to_string(),
74
+ );
75
+ }
76
+ }
77
+ }
78
+ }
79
+ }
80
+
81
+ Statement {
82
+ kind: StatementKind::Spawn {
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
+ }
@@ -1,65 +1,135 @@
1
- use std::collections::HashMap;
2
-
3
- use crate::core::{
4
- lexer::token::{ Token, TokenKind },
5
- parser::{ driver::Parser, statement::{ Statement, StatementKind } },
6
- shared::value::Value,
7
- store::global::GlobalStore,
8
- };
9
-
10
- pub fn parse_synth_token(
11
- parser: &mut Parser,
12
- current_token: Token,
13
- global_store: &mut GlobalStore
14
- ) -> Statement {
15
- parser.advance(); // consume 'synth'
16
-
17
- let Some(synth_token) = parser.previous_clone() else {
18
- return Statement::unknown();
19
- };
20
-
21
- // Expect an identifier (synth waveform)
22
- let Some(identifier_token) = parser.peek_clone() else {
23
- return Statement::error(synth_token, "Expected identifier after 'synth'".to_string());
24
- };
25
-
26
- let synth_waveform = identifier_token.lexeme.clone();
27
-
28
- parser.advance(); // consume identifier
29
-
30
- // Expect synth optional parameters map
31
- let mut parameters = HashMap::new();
32
-
33
- if let Some(params) = parser.parse_map_value() {
34
- // If parameters are provided, we expect a map
35
- if let Value::Map(map) = params {
36
- parameters = map;
37
- } else {
38
- return Statement::error(synth_token, "Expected a map for synth parameters".to_string());
39
- }
40
- } else {
41
- // If no parameters are provided, we can still create the statement with an empty map
42
- parameters = HashMap::new();
43
- }
44
-
45
- Statement {
46
- kind: StatementKind::Synth,
47
- value: Value::Map(
48
- HashMap::from([
49
- ("entity".to_string(), Value::String("synth".to_string())),
50
- (
51
- "value".to_string(),
52
- Value::Map(
53
- HashMap::from([
54
- ("waveform".to_string(), Value::String(synth_waveform)),
55
- ("parameters".to_string(), Value::Map(parameters)),
56
- ])
57
- ),
58
- ),
59
- ])
60
- ),
61
- indent: synth_token.indent,
62
- line: synth_token.line,
63
- column: synth_token.column,
64
- }
65
- }
1
+ use std::collections::HashMap;
2
+
3
+ use devalang_types::Value;
4
+
5
+ use crate::core::{
6
+ lexer::token::Token,
7
+ parser::{
8
+ driver::Parser,
9
+ handler::dot::parse_dot_token,
10
+ statement::{Statement, StatementKind},
11
+ },
12
+ store::global::GlobalStore,
13
+ };
14
+
15
+ pub fn parse_synth_token(
16
+ parser: &mut Parser,
17
+ _current_token: Token,
18
+ _global_store: &mut GlobalStore,
19
+ ) -> Statement {
20
+ parser.advance(); // consume 'synth'
21
+
22
+ let Some(synth_token) = parser.previous_clone() else {
23
+ return Statement::unknown();
24
+ };
25
+
26
+ // Expect a provider/waveform identifier (can be dotted: alias.synth)
27
+ // Also accept a dot-led entity by delegating to the dot parser (e.g. .module.export)
28
+ let synth_waveform = if let Some(first_token) = parser.peek_clone() {
29
+ use crate::core::lexer::token::TokenKind;
30
+
31
+ if first_token.kind == TokenKind::Dot {
32
+ // Parse dot-entity and extract its entity string
33
+ let dot_stmt = parse_dot_token(parser, _global_store);
34
+ // Extract entity if the parsed statement is a Trigger
35
+ match dot_stmt.kind {
36
+ StatementKind::Trigger { entity, .. } => entity,
37
+ _ => String::new(),
38
+ }
39
+ } else {
40
+ if first_token.kind != crate::core::lexer::token::TokenKind::Identifier
41
+ && first_token.kind != crate::core::lexer::token::TokenKind::Number
42
+ && first_token.kind != crate::core::lexer::token::TokenKind::Synth
43
+ {
44
+ return crate::core::parser::statement::error_from_token(
45
+ first_token.clone(),
46
+ "Expected identifier after 'synth'".to_string(),
47
+ );
48
+ }
49
+
50
+ // Collect dotted parts on the same line
51
+ let mut parts: Vec<String> = Vec::new();
52
+ let current_line = first_token.line;
53
+ loop {
54
+ let Some(tok) = parser.peek_clone() else {
55
+ break;
56
+ };
57
+ if tok.line != current_line {
58
+ break;
59
+ }
60
+ match tok.kind {
61
+ crate::core::lexer::token::TokenKind::Identifier
62
+ | crate::core::lexer::token::TokenKind::Number
63
+ | crate::core::lexer::token::TokenKind::Synth => {
64
+ parts.push(tok.lexeme.clone());
65
+ parser.advance();
66
+ // If next isn't a dot on same line, stop
67
+ if let Some(next) = parser.peek_clone() {
68
+ if !(next.line == current_line
69
+ && next.kind == crate::core::lexer::token::TokenKind::Dot)
70
+ {
71
+ break;
72
+ }
73
+ } else {
74
+ break;
75
+ }
76
+ }
77
+ crate::core::lexer::token::TokenKind::Dot => {
78
+ parser.advance();
79
+ }
80
+ _ => break,
81
+ }
82
+ }
83
+
84
+ parts.join(".")
85
+ }
86
+ } else {
87
+ return crate::core::parser::statement::error_from_token(
88
+ synth_token,
89
+ "Expected identifier after 'synth'".to_string(),
90
+ );
91
+ };
92
+
93
+ // Skip formatting before optional parameters map
94
+ while parser.check_token(crate::core::lexer::token::TokenKind::Newline)
95
+ || parser.check_token(crate::core::lexer::token::TokenKind::Indent)
96
+ || parser.check_token(crate::core::lexer::token::TokenKind::Dedent)
97
+ || parser.check_token(crate::core::lexer::token::TokenKind::Whitespace)
98
+ {
99
+ parser.advance();
100
+ }
101
+
102
+ // Expect synth optional parameters map
103
+ let parameters = if let Some(params) = parser.parse_map_value() {
104
+ // If parameters are provided, we expect a map
105
+ if let Value::Map(map) = params {
106
+ map
107
+ } else {
108
+ return crate::core::parser::statement::error_from_token(
109
+ synth_token,
110
+ "Expected a map for synth parameters".to_string(),
111
+ );
112
+ }
113
+ } else {
114
+ // If no parameters are provided, we can still create the statement with an empty map
115
+ HashMap::new()
116
+ };
117
+
118
+ Statement {
119
+ kind: StatementKind::Synth,
120
+ value: Value::Map(HashMap::from([
121
+ ("entity".to_string(), Value::String("synth".to_string())),
122
+ (
123
+ "value".to_string(),
124
+ Value::Map(HashMap::from([
125
+ // Store waveform as identifier to allow resolution from variables/exports
126
+ ("waveform".to_string(), Value::Identifier(synth_waveform)),
127
+ ("parameters".to_string(), Value::Map(parameters)),
128
+ ])),
129
+ ),
130
+ ])),
131
+ indent: synth_token.indent,
132
+ line: synth_token.line,
133
+ column: synth_token.column,
134
+ }
135
+ }