@devaloop/devalang 0.0.1-alpha.15 → 0.0.1-alpha.16-hotfix.0

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