@devaloop/devalang 0.0.1-alpha.14 → 0.0.1-alpha.16

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 (177) hide show
  1. package/.devalang +10 -8
  2. package/.github/workflows/ci.yml +92 -0
  3. package/Cargo.toml +60 -58
  4. package/README.md +32 -15
  5. package/docs/CHANGELOG.md +93 -1
  6. package/docs/CONTRIBUTING.md +101 -1
  7. package/docs/ROADMAP.md +2 -2
  8. package/docs/TODO.md +1 -1
  9. package/examples/automation.deva +42 -0
  10. package/examples/bank.deva +4 -4
  11. package/examples/events.deva +12 -0
  12. package/examples/function.deva +4 -4
  13. package/examples/index.deva +39 -25
  14. package/examples/loop.deva +5 -11
  15. package/examples/pattern.deva +8 -0
  16. package/examples/plugin.deva +16 -0
  17. package/examples/variables.deva +1 -1
  18. package/out-tsc/bin/index.js +51 -7
  19. package/out-tsc/index.js +3 -1
  20. package/out-tsc/scripts/postbuild.js +9 -10
  21. package/out-tsc/scripts/postinstall.js +49 -0
  22. package/package.json +12 -4
  23. package/project-version.json +3 -3
  24. package/rust/cli/bank.rs +462 -456
  25. package/rust/cli/build.rs +252 -199
  26. package/rust/cli/check.rs +221 -180
  27. package/rust/cli/driver.rs +297 -292
  28. package/rust/cli/generator.rs +1 -0
  29. package/rust/cli/init.rs +87 -79
  30. package/rust/cli/install.rs +35 -32
  31. package/rust/cli/login.rs +127 -134
  32. package/rust/cli/mod.rs +13 -11
  33. package/rust/cli/play.rs +1123 -218
  34. package/rust/cli/telemetry.rs +19 -0
  35. package/rust/cli/template.rs +69 -57
  36. package/rust/cli/update.rs +6 -4
  37. package/rust/common/api.rs +5 -8
  38. package/rust/common/cdn.rs +3 -6
  39. package/rust/common/mod.rs +3 -3
  40. package/rust/common/sso.rs +3 -6
  41. package/rust/config/driver.rs +118 -94
  42. package/rust/config/loader.rs +165 -156
  43. package/rust/config/mod.rs +4 -2
  44. package/rust/config/settings.rs +91 -0
  45. package/rust/config/stats.rs +257 -0
  46. package/rust/core/audio/engine.rs +696 -518
  47. package/rust/core/audio/evaluator.rs +263 -31
  48. package/rust/core/audio/interpreter/arrow_call.rs +198 -161
  49. package/rust/core/audio/interpreter/automate.rs +18 -0
  50. package/rust/core/audio/interpreter/call.rs +98 -95
  51. package/rust/core/audio/interpreter/condition.rs +70 -71
  52. package/rust/core/audio/interpreter/driver.rs +487 -198
  53. package/rust/core/audio/interpreter/function.rs +26 -21
  54. package/rust/core/audio/interpreter/let_.rs +38 -19
  55. package/rust/core/audio/interpreter/load.rs +18 -18
  56. package/rust/core/audio/interpreter/loop_.rs +113 -73
  57. package/rust/core/audio/interpreter/mod.rs +14 -13
  58. package/rust/core/audio/interpreter/sleep.rs +27 -30
  59. package/rust/core/audio/interpreter/spawn.rs +105 -102
  60. package/rust/core/audio/interpreter/tempo.rs +19 -16
  61. package/rust/core/audio/interpreter/trigger.rs +239 -210
  62. package/rust/core/audio/loader/mod.rs +1 -1
  63. package/rust/core/audio/loader/trigger.rs +100 -97
  64. package/rust/core/audio/mod.rs +7 -6
  65. package/rust/core/audio/player.rs +64 -64
  66. package/rust/core/audio/renderer.rs +56 -53
  67. package/rust/core/audio/special/easing.rs +189 -0
  68. package/rust/core/audio/special/env.rs +43 -0
  69. package/rust/core/audio/special/math.rs +102 -0
  70. package/rust/core/audio/special/mod.rs +9 -0
  71. package/rust/core/audio/special/modulator.rs +143 -0
  72. package/rust/core/builder/mod.rs +80 -85
  73. package/rust/core/debugger/lexer.rs +27 -27
  74. package/rust/core/debugger/mod.rs +24 -23
  75. package/rust/core/debugger/module.rs +55 -47
  76. package/rust/core/debugger/preprocessor.rs +27 -27
  77. package/rust/core/debugger/store.rs +40 -39
  78. package/rust/core/error/mod.rs +80 -66
  79. package/rust/core/lexer/handler/arrow.rs +82 -31
  80. package/rust/core/lexer/handler/at.rs +21 -21
  81. package/rust/core/lexer/handler/brace.rs +41 -41
  82. package/rust/core/lexer/handler/colon.rs +21 -21
  83. package/rust/core/lexer/handler/comment.rs +30 -30
  84. package/rust/core/lexer/handler/dot.rs +21 -21
  85. package/rust/core/lexer/handler/driver.rs +337 -263
  86. package/rust/core/lexer/handler/identifier.rs +46 -42
  87. package/rust/core/lexer/handler/indent.rs +66 -66
  88. package/rust/core/lexer/handler/mod.rs +16 -16
  89. package/rust/core/lexer/handler/newline.rs +23 -23
  90. package/rust/core/lexer/handler/number.rs +31 -31
  91. package/rust/core/lexer/handler/operator.rs +46 -44
  92. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  93. package/rust/core/lexer/handler/slash.rs +21 -21
  94. package/rust/core/lexer/handler/string.rs +63 -63
  95. package/rust/core/lexer/mod.rs +54 -51
  96. package/rust/core/lexer/token.rs +97 -91
  97. package/rust/core/mod.rs +11 -11
  98. package/rust/core/parser/driver.rs +513 -408
  99. package/rust/core/parser/handler/arrow_call.rs +233 -211
  100. package/rust/core/parser/handler/at.rs +245 -162
  101. package/rust/core/parser/handler/bank.rs +94 -69
  102. package/rust/core/parser/handler/condition.rs +80 -74
  103. package/rust/core/parser/handler/dot.rs +143 -135
  104. package/rust/core/parser/handler/identifier/automate.rs +257 -0
  105. package/rust/core/parser/handler/identifier/call.rs +91 -88
  106. package/rust/core/parser/handler/identifier/emit.rs +66 -0
  107. package/rust/core/parser/handler/identifier/function.rs +100 -92
  108. package/rust/core/parser/handler/identifier/group.rs +85 -75
  109. package/rust/core/parser/handler/identifier/let_.rs +158 -127
  110. package/rust/core/parser/handler/identifier/mod.rs +54 -52
  111. package/rust/core/parser/handler/identifier/on.rs +98 -0
  112. package/rust/core/parser/handler/identifier/print.rs +52 -0
  113. package/rust/core/parser/handler/identifier/sleep.rs +36 -33
  114. package/rust/core/parser/handler/identifier/spawn.rs +91 -88
  115. package/rust/core/parser/handler/identifier/synth.rs +65 -65
  116. package/rust/core/parser/handler/loop_.rs +170 -72
  117. package/rust/core/parser/handler/mod.rs +8 -8
  118. package/rust/core/parser/handler/tempo.rs +53 -47
  119. package/rust/core/parser/mod.rs +4 -4
  120. package/rust/core/parser/statement.rs +142 -108
  121. package/rust/core/plugin/loader.rs +123 -48
  122. package/rust/core/plugin/mod.rs +2 -1
  123. package/rust/core/plugin/runner.rs +296 -0
  124. package/rust/core/preprocessor/loader.rs +515 -326
  125. package/rust/core/preprocessor/mod.rs +4 -4
  126. package/rust/core/preprocessor/module.rs +60 -58
  127. package/rust/core/preprocessor/processor.rs +99 -101
  128. package/rust/core/preprocessor/resolver/bank.rs +51 -49
  129. package/rust/core/preprocessor/resolver/call.rs +100 -100
  130. package/rust/core/preprocessor/resolver/condition.rs +97 -97
  131. package/rust/core/preprocessor/resolver/driver.rs +310 -278
  132. package/rust/core/preprocessor/resolver/function.rs +69 -78
  133. package/rust/core/preprocessor/resolver/group.rs +96 -91
  134. package/rust/core/preprocessor/resolver/let_.rs +32 -28
  135. package/rust/core/preprocessor/resolver/loop_.rs +320 -91
  136. package/rust/core/preprocessor/resolver/mod.rs +15 -15
  137. package/rust/core/preprocessor/resolver/spawn.rs +76 -92
  138. package/rust/core/preprocessor/resolver/synth.rs +56 -50
  139. package/rust/core/preprocessor/resolver/tempo.rs +50 -49
  140. package/rust/core/preprocessor/resolver/trigger.rs +113 -116
  141. package/rust/core/preprocessor/resolver/value.rs +81 -87
  142. package/rust/core/shared/bank.rs +1 -1
  143. package/rust/core/shared/duration.rs +9 -9
  144. package/rust/core/shared/mod.rs +3 -3
  145. package/rust/core/shared/value.rs +35 -32
  146. package/rust/core/store/function.rs +34 -34
  147. package/rust/core/store/global.rs +55 -38
  148. package/rust/core/store/mod.rs +5 -5
  149. package/rust/core/store/variable.rs +37 -34
  150. package/rust/core/utils/mod.rs +2 -2
  151. package/rust/core/utils/path.rs +37 -31
  152. package/rust/core/utils/validation.rs +35 -37
  153. package/rust/installer/addon.rs +84 -80
  154. package/rust/installer/bank.rs +62 -65
  155. package/rust/installer/mod.rs +5 -5
  156. package/rust/installer/plugin.rs +54 -55
  157. package/rust/installer/utils.rs +56 -56
  158. package/rust/lib.rs +156 -164
  159. package/rust/main.rs +250 -145
  160. package/rust/utils/error.rs +200 -0
  161. package/rust/utils/file.rs +38 -35
  162. package/rust/utils/first_usage.rs +76 -0
  163. package/rust/utils/logger.rs +195 -139
  164. package/rust/utils/mod.rs +9 -50
  165. package/rust/utils/signature.rs +19 -17
  166. package/rust/utils/spinner.rs +22 -19
  167. package/rust/utils/telemetry.rs +292 -0
  168. package/rust/utils/watcher.rs +34 -33
  169. package/templates/minimal/README.md +97 -121
  170. package/templates/welcome/README.md +97 -121
  171. package/typescript/bin/index.ts +19 -5
  172. package/typescript/index.ts +3 -1
  173. package/typescript/scripts/postbuild.ts +10 -6
  174. package/typescript/scripts/postinstall.ts +56 -0
  175. package/typescript/scripts/version/bump.ts +0 -1
  176. package/typescript/scripts/version/index.ts +0 -1
  177. package/out-tsc/bin/devalang.exe +0 -0
@@ -1,211 +1,233 @@
1
- use crate::core::{
2
- lexer::token::TokenKind,
3
- parser::{ driver::Parser, statement::{ Statement, StatementKind } },
4
- shared::value::Value,
5
- store::global::GlobalStore,
6
- };
7
-
8
- pub fn parse_arrow_call(parser: &mut Parser, global_store: &mut GlobalStore) -> Statement {
9
- let Some(target_token) = parser.peek_clone() else {
10
- return Statement::unknown();
11
- };
12
-
13
- if target_token.kind != TokenKind::Identifier {
14
- parser.advance(); // consume target token
15
- return Statement::unknown();
16
- }
17
-
18
- let Some(arrow_token) = parser.peek_nth(1).cloned() else {
19
- parser.advance(); // consume arrow token
20
- return Statement {
21
- kind: StatementKind::Unknown,
22
- value: Value::String(target_token.lexeme.clone()),
23
- indent: target_token.indent,
24
- line: target_token.line,
25
- column: target_token.column,
26
- };
27
- };
28
-
29
- if arrow_token.kind != TokenKind::Arrow {
30
- parser.advance(); // consume method token
31
- return Statement {
32
- kind: StatementKind::Unknown,
33
- value: Value::String(target_token.lexeme.clone()),
34
- indent: target_token.indent,
35
- line: target_token.line,
36
- column: target_token.column,
37
- };
38
- }
39
-
40
- // We have a valid arrow call, so we consume the arrow token
41
- let Some(method_token) = parser.peek_nth(2).cloned() else {
42
- parser.advance();
43
- return Statement::unknown();
44
- };
45
-
46
- if method_token.kind != TokenKind::Identifier {
47
- parser.advance();
48
- return Statement::unknown();
49
- }
50
-
51
- // Consume the tokens for target, arrow, and method
52
- parser.advance(); // target
53
- parser.advance(); // ->
54
- parser.advance(); // method
55
-
56
- let mut args = Vec::new();
57
- let mut paren_depth = 0;
58
- let mut map_depth = 0;
59
-
60
- while let Some(token) = parser.peek_clone() {
61
- if token.kind == TokenKind::Newline || token.kind == TokenKind::EOF {
62
- break;
63
- }
64
- if token.kind == TokenKind::LParen {
65
- paren_depth += 1;
66
- }
67
- if token.kind == TokenKind::RParen {
68
- if paren_depth > 0 {
69
- paren_depth -= 1;
70
- parser.advance();
71
- if paren_depth == 0 {
72
- break;
73
- }
74
- continue;
75
- } else {
76
- break;
77
- }
78
- }
79
- if token.kind == TokenKind::LBrace {
80
- map_depth += 1;
81
- }
82
- if token.kind == TokenKind::RBrace {
83
- if map_depth > 0 {
84
- map_depth -= 1;
85
- parser.advance();
86
- if map_depth == 0 {
87
- continue;
88
- }
89
- continue;
90
- } else {
91
- break;
92
- }
93
- }
94
-
95
- parser.advance();
96
-
97
- let value = match token.kind {
98
- TokenKind::Identifier => Value::Identifier(token.lexeme.clone()),
99
- TokenKind::String => Value::String(token.lexeme.clone()),
100
- TokenKind::Number => Value::Number(token.lexeme.parse::<f32>().unwrap_or(0.0)),
101
- TokenKind::LBrace => {
102
- // Handle map literal
103
- let mut map = std::collections::HashMap::new();
104
- while let Some(inner_token) = parser.peek_clone() {
105
- if inner_token.kind == TokenKind::RBrace {
106
- parser.advance(); // consume RBrace
107
- break;
108
- }
109
- if inner_token.kind == TokenKind::Newline || inner_token.kind == TokenKind::EOF {
110
- break;
111
- }
112
- parser.advance(); // consume key token
113
- let key = inner_token.lexeme.clone();
114
- if let Some(colon_token) = parser.peek_clone() {
115
- if colon_token.kind == TokenKind::Colon {
116
- parser.advance(); // consume colon
117
- if let Some(value_token) = parser.peek_clone() {
118
- parser.advance(); // consume value token
119
- let value = match value_token.kind {
120
- TokenKind::Identifier => {
121
- // Interpret bare true/false as booleans
122
- if value_token.lexeme == "true" {
123
- Value::Boolean(true)
124
- } else if value_token.lexeme == "false" {
125
- Value::Boolean(false)
126
- } else {
127
- Value::Identifier(value_token.lexeme.clone())
128
- }
129
- },
130
- TokenKind::String => Value::String(value_token.lexeme.clone()),
131
- TokenKind::Number => {
132
- // Support decimals (e.g., 0.8) and beats (e.g., 1/4)
133
- if let Some(TokenKind::Slash) = parser.peek_kind() {
134
- // Beat fraction
135
- parser.advance(); // consume '/'
136
- if let Some(denominator_token) = parser.peek_clone() {
137
- if denominator_token.kind == TokenKind::Number {
138
- parser.advance(); // consume denominator
139
- let denominator = denominator_token.lexeme.clone();
140
- Value::Beat(format!("{}/{}", value_token.lexeme, denominator))
141
- } else {
142
- Value::Unknown
143
- }
144
- } else {
145
- Value::Unknown
146
- }
147
- } else if let Some(next) = parser.peek_clone() {
148
- // Decimal number handling: NUMBER '.' NUMBER -> f32
149
- if next.kind == TokenKind::Dot {
150
- // consume '.'
151
- parser.advance();
152
- if let Some(after_dot) = parser.peek_clone() {
153
- if after_dot.kind == TokenKind::Number {
154
- parser.advance(); // consume fractional digits
155
- let combined = format!("{}.{}", value_token.lexeme, after_dot.lexeme);
156
- Value::Number(combined.parse::<f32>().unwrap_or(0.0))
157
- } else {
158
- // Lone dot without number, fallback to integer part
159
- Value::Number(value_token.lexeme.parse::<f32>().unwrap_or(0.0))
160
- }
161
- } else {
162
- Value::Number(value_token.lexeme.parse::<f32>().unwrap_or(0.0))
163
- }
164
- } else {
165
- // Regular integer number
166
- Value::Number(value_token.lexeme.parse::<f32>().unwrap_or(0.0))
167
- }
168
- } else {
169
- Value::Number(value_token.lexeme.parse::<f32>().unwrap_or(0.0))
170
- }
171
- }
172
- TokenKind::Boolean =>
173
- Value::Boolean(
174
- value_token.lexeme.parse::<bool>().unwrap_or(false)
175
- ),
176
- _ => Value::Unknown,
177
- };
178
- map.insert(key, value);
179
- }
180
- }
181
- }
182
- }
183
- Value::Map(map)
184
- }
185
- _ => Value::Unknown,
186
- };
187
-
188
- args.push(value);
189
-
190
- // Stop if we reach the end of the statement
191
- if
192
- paren_depth == 0 &&
193
- map_depth == 0 &&
194
- (token.kind == TokenKind::RParen || token.kind == TokenKind::RBrace)
195
- {
196
- break;
197
- }
198
- }
199
-
200
- Statement {
201
- kind: StatementKind::ArrowCall {
202
- target: target_token.lexeme.clone(),
203
- method: method_token.lexeme.clone(),
204
- args,
205
- },
206
- value: Value::Null,
207
- indent: target_token.indent,
208
- line: target_token.line,
209
- column: target_token.column,
210
- }
211
- }
1
+ use crate::core::{
2
+ lexer::token::TokenKind,
3
+ parser::{
4
+ driver::Parser,
5
+ statement::{Statement, StatementKind},
6
+ },
7
+ shared::value::Value,
8
+ store::global::GlobalStore,
9
+ };
10
+
11
+ fn parse_map_literal(parser: &mut Parser) -> Value {
12
+ // Assumes '{' has already been consumed by caller
13
+ let mut map = std::collections::HashMap::new();
14
+ loop {
15
+ let Some(inner_token) = parser.peek_clone() else {
16
+ break;
17
+ };
18
+
19
+ match inner_token.kind {
20
+ TokenKind::RBrace => {
21
+ parser.advance(); // consume '}'
22
+ break;
23
+ }
24
+ TokenKind::Newline | TokenKind::Comma => {
25
+ parser.advance();
26
+ continue;
27
+ }
28
+ _ => {}
29
+ }
30
+
31
+ // Key
32
+ parser.advance();
33
+ let key = inner_token.lexeme.clone();
34
+
35
+ // Expect ':'
36
+ if let Some(colon_token) = parser.peek_clone() {
37
+ if colon_token.kind == TokenKind::Colon {
38
+ parser.advance(); // consume ':'
39
+
40
+ // Value
41
+ if let Some(value_token) = parser.peek_clone() {
42
+ match value_token.kind {
43
+ TokenKind::LBrace => {
44
+ parser.advance(); // consume '{'
45
+ let nested = parse_map_literal(parser);
46
+ map.insert(key, nested);
47
+ }
48
+ TokenKind::Identifier => {
49
+ parser.advance();
50
+ let v = if value_token.lexeme == "true" {
51
+ Value::Boolean(true)
52
+ } else if value_token.lexeme == "false" {
53
+ Value::Boolean(false)
54
+ } else {
55
+ Value::Identifier(value_token.lexeme.clone())
56
+ };
57
+ map.insert(key, v);
58
+ }
59
+ TokenKind::String => {
60
+ parser.advance();
61
+ map.insert(key, Value::String(value_token.lexeme.clone()));
62
+ }
63
+ TokenKind::Number => {
64
+ parser.advance();
65
+ // Beat fraction support: NUMBER '/' NUMBER
66
+ if let Some(TokenKind::Slash) = parser.peek_kind() {
67
+ parser.advance(); // '/'
68
+ if let Some(den) = parser.peek_clone() {
69
+ if den.kind == TokenKind::Number {
70
+ parser.advance();
71
+ let beat = format!("{}/{}", value_token.lexeme, den.lexeme);
72
+ map.insert(key, Value::Beat(beat));
73
+ continue;
74
+ }
75
+ }
76
+ }
77
+ // Decimal support NUMBER '.' NUMBER
78
+ if let Some(next) = parser.peek_clone() {
79
+ if next.kind == TokenKind::Dot {
80
+ parser.advance(); // '.'
81
+ if let Some(after) = parser.peek_clone() {
82
+ if after.kind == TokenKind::Number {
83
+ parser.advance();
84
+ let combined =
85
+ format!("{}.{}", value_token.lexeme, after.lexeme);
86
+ map.insert(
87
+ key,
88
+ Value::Number(
89
+ combined.parse::<f32>().unwrap_or(0.0),
90
+ ),
91
+ );
92
+ continue;
93
+ }
94
+ }
95
+ }
96
+ }
97
+ map.insert(
98
+ key,
99
+ Value::Number(value_token.lexeme.parse::<f32>().unwrap_or(0.0)),
100
+ );
101
+ }
102
+ TokenKind::Boolean => {
103
+ parser.advance();
104
+ map.insert(
105
+ key,
106
+ Value::Boolean(value_token.lexeme.parse::<bool>().unwrap_or(false)),
107
+ );
108
+ }
109
+ _ => {
110
+ // Unknown value type, consume and store Unknown
111
+ parser.advance();
112
+ map.insert(key, Value::Unknown);
113
+ }
114
+ }
115
+ }
116
+ }
117
+ }
118
+ }
119
+ Value::Map(map)
120
+ }
121
+
122
+ pub fn parse_arrow_call(parser: &mut Parser, _global_store: &mut GlobalStore) -> Statement {
123
+ let Some(target_token) = parser.peek_clone() else {
124
+ return Statement::unknown();
125
+ };
126
+
127
+ if target_token.kind != TokenKind::Identifier {
128
+ parser.advance(); // consume target token
129
+ return Statement::unknown_from_token(&target_token);
130
+ }
131
+
132
+ let Some(arrow_token) = parser.peek_nth(1).cloned() else {
133
+ parser.advance(); // consume arrow token
134
+ return Statement::unknown_from_token(&target_token);
135
+ };
136
+
137
+ if arrow_token.kind != TokenKind::Arrow {
138
+ parser.advance(); // consume method token
139
+ return Statement::unknown_from_token(&target_token);
140
+ }
141
+
142
+ // We have a valid arrow call, so we consume the arrow token
143
+ let Some(method_token) = parser.peek_nth(2).cloned() else {
144
+ parser.advance();
145
+ return Statement::unknown_from_token(&target_token);
146
+ };
147
+
148
+ if method_token.kind != TokenKind::Identifier {
149
+ parser.advance();
150
+ return Statement::unknown_from_token(&method_token);
151
+ }
152
+
153
+ // Consume the tokens for target, arrow, and method
154
+ parser.advance(); // target
155
+ parser.advance(); // ->
156
+ parser.advance(); // method
157
+
158
+ let mut args = Vec::new();
159
+ let mut paren_depth = 0;
160
+ let mut map_depth = 0;
161
+
162
+ while let Some(token) = parser.peek_clone() {
163
+ if token.kind == TokenKind::Newline || token.kind == TokenKind::EOF {
164
+ break;
165
+ }
166
+ if token.kind == TokenKind::LParen {
167
+ paren_depth += 1;
168
+ }
169
+ if token.kind == TokenKind::RParen {
170
+ if paren_depth > 0 {
171
+ paren_depth -= 1;
172
+ parser.advance();
173
+ if paren_depth == 0 {
174
+ break;
175
+ }
176
+ continue;
177
+ } else {
178
+ break;
179
+ }
180
+ }
181
+ if token.kind == TokenKind::LBrace {
182
+ map_depth += 1;
183
+ }
184
+ if token.kind == TokenKind::RBrace {
185
+ if map_depth > 0 {
186
+ map_depth -= 1;
187
+ parser.advance();
188
+ if map_depth == 0 {
189
+ continue;
190
+ }
191
+ continue;
192
+ } else {
193
+ break;
194
+ }
195
+ }
196
+
197
+ parser.advance();
198
+
199
+ let value = match token.kind {
200
+ TokenKind::Identifier => Value::Identifier(token.lexeme.clone()),
201
+ TokenKind::String => Value::String(token.lexeme.clone()),
202
+ TokenKind::Number => Value::Number(token.lexeme.parse::<f32>().unwrap_or(0.0)),
203
+ TokenKind::LBrace => {
204
+ // Handle map literal (supports nested maps)
205
+ let map_val = parse_map_literal(parser);
206
+ // We consumed the matching '}', so outer map_depth should be decremented
207
+ // if the caller tracks it.
208
+ map_val
209
+ }
210
+ _ => Value::Unknown,
211
+ };
212
+
213
+ args.push(value);
214
+
215
+ // Stop if we reach the end of the statement
216
+ if paren_depth == 0 && (token.kind == TokenKind::RParen || token.kind == TokenKind::RBrace)
217
+ {
218
+ break;
219
+ }
220
+ }
221
+
222
+ Statement {
223
+ kind: StatementKind::ArrowCall {
224
+ target: target_token.lexeme.clone(),
225
+ method: method_token.lexeme.clone(),
226
+ args,
227
+ },
228
+ value: Value::Null,
229
+ indent: target_token.indent,
230
+ line: target_token.line,
231
+ column: target_token.column,
232
+ }
233
+ }