@devaloop/devalang 0.0.1-beta.1 → 0.0.1-beta.2
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.
- package/.devalang +9 -10
- package/Cargo.toml +5 -4
- package/README.md +7 -5
- package/docs/CHANGELOG.md +42 -0
- package/docs/ROADMAP.md +5 -1
- package/docs/TODO.md +3 -14
- package/examples/bus.deva +10 -0
- package/examples/effect.deva +2 -0
- package/examples/filter.deva +11 -0
- package/examples/lfo.deva +9 -0
- package/examples/synth.deva +11 -1
- package/examples/synth_types.deva +17 -0
- package/out-tsc/core/functions/index.d.ts +5 -0
- package/out-tsc/core/functions/index.js +11 -0
- package/out-tsc/pkg/devalang_core.d.ts +2 -0
- package/out-tsc/pkg/devalang_core.js +17 -2
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -7
- package/package.json +1 -1
- package/project-version.json +3 -3
- package/rust/cli/bank/api.rs +122 -122
- package/rust/cli/bank/commands.rs +33 -2
- package/rust/cli/bank/mod.rs +29 -29
- package/rust/cli/build/commands.rs +53 -3
- package/rust/cli/build/mod.rs +2 -2
- package/rust/cli/build/process.rs +26 -7
- package/rust/cli/check/mod.rs +2 -2
- package/rust/cli/discover/commands.rs +253 -253
- package/rust/cli/discover/config.rs +111 -111
- package/rust/cli/discover/fs.rs +19 -19
- package/rust/cli/discover/install.rs +103 -103
- package/rust/cli/discover/metadata.rs +48 -48
- package/rust/cli/discover/mod.rs +5 -5
- package/rust/cli/install/addon.rs +118 -118
- package/rust/cli/install/bank.rs +22 -3
- package/rust/cli/install/commands.rs +35 -35
- package/rust/cli/install/mod.rs +4 -4
- package/rust/cli/install/plugin.rs +80 -61
- package/rust/cli/login/commands.rs +124 -124
- package/rust/cli/mod.rs +12 -12
- package/rust/cli/parser.rs +46 -1
- package/rust/cli/play/commands.rs +71 -20
- package/rust/cli/play/mod.rs +5 -5
- package/rust/cli/play/process.rs +14 -5
- package/rust/cli/play/realtime.rs +91 -91
- package/rust/cli/telemetry/commands.rs +22 -22
- package/rust/cli/telemetry/event_creator.rs +80 -80
- package/rust/cli/telemetry/mod.rs +3 -3
- package/rust/cli/telemetry/send.rs +51 -51
- package/rust/cli/template/commands.rs +69 -69
- package/rust/config/driver.rs +112 -103
- package/rust/config/mod.rs +3 -3
- package/rust/config/ops.rs +26 -26
- package/rust/config/settings.rs +101 -101
- package/rust/core/audio/engine/driver.rs +220 -0
- package/rust/core/audio/engine/export.rs +169 -0
- package/rust/core/audio/engine/helpers.rs +178 -170
- package/rust/core/audio/engine/mod.rs +51 -2
- package/rust/core/audio/engine/notes/dsp.rs +85 -0
- package/rust/core/audio/engine/notes/mod.rs +44 -0
- package/rust/core/audio/engine/notes/params.rs +294 -0
- package/rust/core/audio/engine/sample/insert.rs +199 -0
- package/rust/core/audio/engine/sample/mod.rs +40 -0
- package/rust/core/audio/engine/sample/padding.rs +170 -0
- package/rust/core/audio/evaluator/condition.rs +61 -0
- package/rust/core/audio/evaluator/mod.rs +9 -0
- package/rust/core/audio/{evaluator.rs → evaluator/numeric.rs} +1 -159
- package/rust/core/audio/evaluator/rhs.rs +16 -0
- package/rust/core/audio/evaluator/string_expr.rs +94 -0
- package/rust/core/audio/interpreter/driver.rs +55 -23
- package/rust/core/audio/interpreter/mod.rs +1 -13
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
- package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
- package/rust/core/audio/interpreter/{automate.rs → statements/automate.rs} +16 -18
- package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +5 -4
- package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +2 -1
- package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +2 -4
- package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +2 -4
- package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +2 -4
- package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +2 -1
- package/rust/core/audio/interpreter/statements/mod.rs +12 -0
- package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
- package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +3 -2
- package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
- package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +1 -1
- package/rust/core/audio/loader/trigger.rs +2 -1
- package/rust/core/audio/mod.rs +6 -7
- package/rust/core/audio/player.rs +70 -70
- package/rust/core/audio/special/easing.rs +189 -189
- package/rust/core/audio/special/env.rs +45 -45
- package/rust/core/audio/special/math.rs +134 -134
- package/rust/core/audio/special/mod.rs +9 -9
- package/rust/core/audio/special/modulator.rs +143 -143
- package/rust/core/builder/mod.rs +45 -2
- package/rust/core/debugger/lexer.rs +27 -27
- package/rust/core/debugger/{module.rs → logs.rs} +3 -6
- package/rust/core/debugger/mod.rs +30 -30
- package/rust/core/debugger/preprocessor.rs +27 -27
- package/rust/core/debugger/store.rs +2 -4
- package/rust/core/error/mod.rs +269 -269
- package/rust/core/lexer/driver.rs +59 -61
- package/rust/core/lexer/handler/arrow.rs +82 -82
- package/rust/core/lexer/handler/at.rs +21 -21
- package/rust/core/lexer/handler/brace.rs +41 -41
- package/rust/core/lexer/handler/colon.rs +21 -21
- package/rust/core/lexer/handler/comment.rs +30 -30
- package/rust/core/lexer/handler/dot.rs +21 -21
- package/rust/core/lexer/handler/driver.rs +337 -337
- package/rust/core/lexer/handler/identifier.rs +47 -47
- package/rust/core/lexer/handler/indent.rs +66 -66
- package/rust/core/lexer/handler/mod.rs +15 -15
- package/rust/core/lexer/handler/newline.rs +23 -23
- package/rust/core/lexer/handler/number.rs +31 -31
- package/rust/core/lexer/handler/operator.rs +46 -46
- package/rust/core/lexer/handler/parenthesis.rs +41 -41
- package/rust/core/lexer/handler/slash.rs +21 -21
- package/rust/core/lexer/handler/string.rs +63 -63
- package/rust/core/lexer/mod.rs +3 -3
- package/rust/core/mod.rs +0 -1
- package/rust/core/parser/driver/block.rs +111 -0
- package/rust/core/parser/driver/cursor.rs +82 -0
- package/rust/core/parser/driver/driver_impl.rs +139 -0
- package/rust/core/parser/driver/mod.rs +6 -0
- package/rust/core/parser/driver/parse_array.rs +120 -0
- package/rust/core/parser/driver/parse_map.rs +223 -0
- package/rust/core/parser/driver/parser.rs +160 -0
- package/rust/core/parser/handler/arrow_call.rs +28 -4
- package/rust/core/parser/handler/at.rs +279 -279
- package/rust/core/parser/handler/bank.rs +104 -104
- package/rust/core/parser/handler/condition.rs +83 -83
- package/rust/core/parser/handler/dot.rs +148 -148
- package/rust/core/parser/handler/identifier/automate.rs +254 -254
- package/rust/core/parser/handler/identifier/call.rs +91 -91
- package/rust/core/parser/handler/identifier/emit.rs +70 -70
- package/rust/core/parser/handler/identifier/function.rs +113 -113
- package/rust/core/parser/handler/identifier/group.rs +89 -89
- package/rust/core/parser/handler/identifier/let_.rs +173 -173
- package/rust/core/parser/handler/identifier/mod.rs +55 -55
- package/rust/core/parser/handler/identifier/on.rs +107 -107
- package/rust/core/parser/handler/identifier/print.rs +49 -49
- package/rust/core/parser/handler/identifier/sleep.rs +96 -43
- package/rust/core/parser/handler/identifier/spawn.rs +91 -91
- package/rust/core/parser/handler/identifier/synth.rs +135 -135
- package/rust/core/parser/handler/loop_.rs +194 -194
- package/rust/core/parser/handler/mod.rs +9 -9
- package/rust/core/parser/handler/pattern.rs +1 -1
- package/rust/core/parser/handler/tempo.rs +105 -57
- package/rust/core/parser/statement.rs +10 -11
- package/rust/core/plugin/loader.rs +1 -1
- package/rust/core/plugin/mod.rs +2 -2
- package/rust/core/plugin/runner/mod.rs +11 -0
- package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +297 -347
- package/rust/core/plugin/runner/wasm32.rs +43 -0
- package/rust/core/preprocessor/loader/inject.rs +278 -0
- package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
- package/rust/core/preprocessor/loader/mod.rs +235 -0
- package/rust/core/preprocessor/module.rs +2 -7
- package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +6 -13
- package/rust/core/preprocessor/processor/mod.rs +1 -0
- package/rust/core/preprocessor/resolver/bank.rs +49 -49
- package/rust/core/preprocessor/resolver/call.rs +124 -124
- package/rust/core/preprocessor/resolver/condition.rs +95 -95
- package/rust/core/preprocessor/resolver/driver.rs +324 -324
- package/rust/core/preprocessor/resolver/function.rs +2 -2
- package/rust/core/preprocessor/resolver/group.rs +46 -18
- package/rust/core/preprocessor/resolver/let_.rs +32 -32
- package/rust/core/preprocessor/resolver/loop_.rs +318 -318
- package/rust/core/preprocessor/resolver/mod.rs +16 -16
- package/rust/core/preprocessor/resolver/pattern.rs +83 -83
- package/rust/core/preprocessor/resolver/spawn.rs +99 -99
- package/rust/core/preprocessor/resolver/synth.rs +54 -54
- package/rust/core/preprocessor/resolver/tempo.rs +48 -48
- package/rust/core/preprocessor/resolver/trigger.rs +116 -116
- package/rust/core/preprocessor/resolver/value.rs +176 -176
- package/rust/core/store/global.rs +2 -6
- package/rust/core/store/mod.rs +1 -5
- package/rust/lib.rs +18 -3
- package/rust/main.rs +27 -3
- package/rust/types/Cargo.toml +1 -1
- package/rust/types/src/addons.rs +55 -55
- package/rust/types/src/config.rs +84 -74
- package/rust/types/src/lib.rs +15 -12
- package/rust/types/src/plugin.rs +20 -0
- package/rust/types/src/store.rs +139 -0
- package/rust/types/src/telemetry.rs +85 -85
- package/rust/utils/Cargo.toml +2 -2
- package/rust/utils/src/file.rs +94 -94
- package/rust/utils/src/first_usage.rs +97 -97
- package/rust/utils/src/lib.rs +9 -9
- package/rust/utils/src/logger.rs +200 -200
- package/rust/utils/src/path.rs +129 -88
- package/rust/utils/src/signature.rs +41 -41
- package/rust/utils/src/spinner.rs +20 -20
- package/rust/utils/src/version.rs +27 -27
- package/rust/utils/src/watcher.rs +46 -46
- package/rust/web/api.rs +5 -5
- package/rust/web/cdn.rs +34 -34
- package/rust/web/mod.rs +3 -3
- package/tests/integration.rs +21 -21
- package/typescript/core/functions/index.ts +11 -0
- package/typescript/pkg/devalang_core.ts +20 -4
- package/rust/core/audio/engine/sample.rs +0 -366
- package/rust/core/audio/engine/synth.rs +0 -325
- package/rust/core/audio/interpreter/arrow_call.rs +0 -311
- package/rust/core/audio/renderer.rs +0 -54
- package/rust/core/parser/driver.rs +0 -584
- package/rust/core/preprocessor/loader.rs +0 -637
- package/rust/core/store/export.rs +0 -28
- package/rust/core/store/function.rs +0 -40
- package/rust/core/store/import.rs +0 -28
- package/rust/core/store/variable.rs +0 -51
- package/rust/core/utils/mod.rs +0 -1
- package/rust/core/utils/path.rs +0 -37
|
@@ -0,0 +1,223 @@
|
|
|
1
|
+
use crate::core::lexer::token::TokenKind;
|
|
2
|
+
use devalang_types::Value;
|
|
3
|
+
|
|
4
|
+
pub fn parse_map_value(parser: &mut crate::core::parser::driver::parser::Parser) -> Option<Value> {
|
|
5
|
+
let logger = devalang_utils::logger::Logger::new();
|
|
6
|
+
use devalang_utils::logger::LogLevel;
|
|
7
|
+
if !parser.match_token(TokenKind::LBrace) {
|
|
8
|
+
return None;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
let mut map = std::collections::HashMap::new();
|
|
12
|
+
|
|
13
|
+
while !parser.check_token(TokenKind::RBrace) && !parser.is_eof() {
|
|
14
|
+
// Skip separators and formatting before the key
|
|
15
|
+
while parser.check_token(TokenKind::Newline)
|
|
16
|
+
|| parser.check_token(TokenKind::Whitespace)
|
|
17
|
+
|| parser.check_token(TokenKind::Indent)
|
|
18
|
+
|| parser.check_token(TokenKind::Dedent)
|
|
19
|
+
|| parser.check_token(TokenKind::Comma)
|
|
20
|
+
{
|
|
21
|
+
parser.advance();
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
// Check if we are at the closing brace of the map
|
|
25
|
+
if parser.check_token(TokenKind::RBrace) {
|
|
26
|
+
break;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
let key = if let Some(token) = parser.advance() {
|
|
30
|
+
match token.kind {
|
|
31
|
+
TokenKind::Whitespace
|
|
32
|
+
| TokenKind::Indent
|
|
33
|
+
| TokenKind::Dedent
|
|
34
|
+
| TokenKind::Newline => {
|
|
35
|
+
continue;
|
|
36
|
+
}
|
|
37
|
+
_ => token.lexeme.clone(),
|
|
38
|
+
}
|
|
39
|
+
} else {
|
|
40
|
+
break;
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
// Skip newlines and whitespace before colon
|
|
44
|
+
while parser.check_token(TokenKind::Newline) || parser.check_token(TokenKind::Whitespace) {
|
|
45
|
+
parser.advance();
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
if !parser.match_token(TokenKind::Colon) {
|
|
49
|
+
logger.log_message(
|
|
50
|
+
LogLevel::Error,
|
|
51
|
+
&format!("Expected ':' after map key '{}'", key),
|
|
52
|
+
);
|
|
53
|
+
break;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// Skip separators and formatting before value
|
|
57
|
+
while parser.check_token(TokenKind::Newline)
|
|
58
|
+
|| parser.check_token(TokenKind::Whitespace)
|
|
59
|
+
|| parser.check_token(TokenKind::Indent)
|
|
60
|
+
|| parser.check_token(TokenKind::Dedent)
|
|
61
|
+
|| parser.check_token(TokenKind::Comma)
|
|
62
|
+
{
|
|
63
|
+
parser.advance();
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
let value = if let Some(token) = parser.peek_clone() {
|
|
67
|
+
match token.kind {
|
|
68
|
+
TokenKind::String => {
|
|
69
|
+
parser.advance();
|
|
70
|
+
Value::String(token.lexeme.clone())
|
|
71
|
+
}
|
|
72
|
+
TokenKind::Number => {
|
|
73
|
+
// Handle number, decimal number and optional fraction form (e.g., 1/4)
|
|
74
|
+
let mut number_str = token.lexeme.clone();
|
|
75
|
+
parser.advance(); // consume the first number
|
|
76
|
+
|
|
77
|
+
// decimal support: number '.' number
|
|
78
|
+
if let Some(dot_token) = parser.peek_clone() {
|
|
79
|
+
if dot_token.kind == TokenKind::Dot {
|
|
80
|
+
parser.advance(); // consume the dot
|
|
81
|
+
|
|
82
|
+
if let Some(decimal_token) = parser.peek_clone() {
|
|
83
|
+
if decimal_token.kind == TokenKind::Number {
|
|
84
|
+
parser.advance(); // consume the number after the dot
|
|
85
|
+
number_str.push('.');
|
|
86
|
+
number_str.push_str(&decimal_token.lexeme);
|
|
87
|
+
} else {
|
|
88
|
+
logger.log_message(
|
|
89
|
+
LogLevel::Error,
|
|
90
|
+
&format!(
|
|
91
|
+
"Expected number after dot, got {:?}",
|
|
92
|
+
decimal_token
|
|
93
|
+
),
|
|
94
|
+
);
|
|
95
|
+
return Some(Value::Null);
|
|
96
|
+
}
|
|
97
|
+
} else {
|
|
98
|
+
logger.log_message(
|
|
99
|
+
LogLevel::Error,
|
|
100
|
+
"Expected number after dot, but reached EOF",
|
|
101
|
+
);
|
|
102
|
+
return Some(Value::Null);
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// Fraction support: number '/' number -> Duration::Beat("num/den")
|
|
108
|
+
if let Some(slash_tok) = parser.peek_clone() {
|
|
109
|
+
if slash_tok.kind == TokenKind::Slash {
|
|
110
|
+
// consume '/'
|
|
111
|
+
parser.advance();
|
|
112
|
+
if let Some(den_tok) = parser.peek_clone() {
|
|
113
|
+
match den_tok.kind {
|
|
114
|
+
TokenKind::Number | TokenKind::Identifier => {
|
|
115
|
+
let frac = format!("{}/{}", number_str, den_tok.lexeme);
|
|
116
|
+
parser.advance();
|
|
117
|
+
return Some(Value::Duration(
|
|
118
|
+
devalang_types::Duration::Beat(frac),
|
|
119
|
+
));
|
|
120
|
+
}
|
|
121
|
+
_ => {
|
|
122
|
+
logger.log_message(
|
|
123
|
+
LogLevel::Error,
|
|
124
|
+
&format!(
|
|
125
|
+
"Expected number or identifier after '/', got {:?}",
|
|
126
|
+
den_tok
|
|
127
|
+
),
|
|
128
|
+
);
|
|
129
|
+
return Some(Value::Null);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
} else {
|
|
133
|
+
logger.log_message(
|
|
134
|
+
LogLevel::Error,
|
|
135
|
+
"Expected denominator after '/', but reached EOF",
|
|
136
|
+
);
|
|
137
|
+
return Some(Value::Null);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
|
|
142
|
+
Value::Number(number_str.parse::<f32>().unwrap_or(0.0))
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
TokenKind::Identifier => {
|
|
146
|
+
// Support dotted identifiers in map values: alias.param or nested
|
|
147
|
+
let current_line = token.line;
|
|
148
|
+
let mut parts: Vec<String> = vec![token.lexeme.clone()];
|
|
149
|
+
parser.advance();
|
|
150
|
+
loop {
|
|
151
|
+
let Some(next) = parser.peek_clone() else {
|
|
152
|
+
break;
|
|
153
|
+
};
|
|
154
|
+
if next.line != current_line {
|
|
155
|
+
break;
|
|
156
|
+
}
|
|
157
|
+
if next.kind == TokenKind::Dot {
|
|
158
|
+
// Consume '.' and the following identifier/number on same line
|
|
159
|
+
parser.advance(); // dot
|
|
160
|
+
if let Some(id2) = parser.peek_clone() {
|
|
161
|
+
if id2.line == current_line
|
|
162
|
+
&& (id2.kind == TokenKind::Identifier
|
|
163
|
+
|| id2.kind == TokenKind::Number)
|
|
164
|
+
{
|
|
165
|
+
parts.push(id2.lexeme.clone());
|
|
166
|
+
parser.advance(); // consume part
|
|
167
|
+
continue;
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
break;
|
|
171
|
+
} else {
|
|
172
|
+
break;
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
Value::Identifier(parts.join("."))
|
|
176
|
+
}
|
|
177
|
+
TokenKind::LBracket => {
|
|
178
|
+
// Allow arrays as map values
|
|
179
|
+
if let Some(v) =
|
|
180
|
+
crate::core::parser::driver::parse_array::parse_array_value(parser)
|
|
181
|
+
{
|
|
182
|
+
v
|
|
183
|
+
} else {
|
|
184
|
+
Value::Null
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
TokenKind::LBrace => {
|
|
188
|
+
// Allow inline nested maps as map values
|
|
189
|
+
if let Some(v) = parse_map_value(parser) {
|
|
190
|
+
v
|
|
191
|
+
} else {
|
|
192
|
+
Value::Null
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
_ => {
|
|
196
|
+
logger.log_message(
|
|
197
|
+
LogLevel::Error,
|
|
198
|
+
&format!("Unexpected token in map value: {:?}", token),
|
|
199
|
+
);
|
|
200
|
+
Value::Null
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
} else {
|
|
204
|
+
Value::Null
|
|
205
|
+
};
|
|
206
|
+
|
|
207
|
+
map.insert(key, value);
|
|
208
|
+
|
|
209
|
+
// Optionally skip a trailing comma after the value
|
|
210
|
+
while parser.check_token(TokenKind::Comma)
|
|
211
|
+
|| parser.check_token(TokenKind::Whitespace)
|
|
212
|
+
|| parser.check_token(TokenKind::Newline)
|
|
213
|
+
{
|
|
214
|
+
parser.advance();
|
|
215
|
+
}
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
if !parser.match_token(TokenKind::RBrace) {
|
|
219
|
+
logger.log_message(LogLevel::Error, "Expected '}' at end of map");
|
|
220
|
+
}
|
|
221
|
+
|
|
222
|
+
Some(Value::Map(map))
|
|
223
|
+
}
|
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
use crate::core::{
|
|
2
|
+
lexer::token::{Token, TokenKind},
|
|
3
|
+
parser::statement::Statement,
|
|
4
|
+
store::global::GlobalStore,
|
|
5
|
+
};
|
|
6
|
+
use devalang_types::Value;
|
|
7
|
+
|
|
8
|
+
#[derive(Debug, Clone, PartialEq)]
|
|
9
|
+
pub struct Parser {
|
|
10
|
+
pub resolve_modules: bool,
|
|
11
|
+
pub tokens: Vec<Token>,
|
|
12
|
+
pub token_index: usize,
|
|
13
|
+
pub current_module: String,
|
|
14
|
+
pub previous: Option<Token>,
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
impl Default for Parser {
|
|
18
|
+
fn default() -> Self {
|
|
19
|
+
Self::new()
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
impl Parser {
|
|
24
|
+
pub fn new() -> Self {
|
|
25
|
+
Parser {
|
|
26
|
+
resolve_modules: false,
|
|
27
|
+
tokens: Vec::new(),
|
|
28
|
+
token_index: 0,
|
|
29
|
+
current_module: String::new(),
|
|
30
|
+
previous: None,
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
pub fn set_current_module(&mut self, module_path: String) {
|
|
35
|
+
self.current_module = module_path;
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
pub fn advance(&mut self) -> Option<&Token> {
|
|
39
|
+
crate::core::parser::driver::cursor::advance_impl(self)
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
pub fn peek_is(&self, expected: &str) -> bool {
|
|
43
|
+
crate::core::parser::driver::cursor::peek_is_impl(self, expected)
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
pub fn peek_nth(&self, n: usize) -> Option<&Token> {
|
|
47
|
+
crate::core::parser::driver::cursor::peek_nth_impl(self, n)
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
pub fn peek_nth_kind(&self, n: usize) -> Option<TokenKind> {
|
|
51
|
+
crate::core::parser::driver::cursor::peek_nth_kind_impl(self, n)
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
pub fn advance_if(&mut self, kind: TokenKind) -> bool {
|
|
55
|
+
crate::core::parser::driver::cursor::advance_if_impl(self, kind)
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
pub fn match_token(&mut self, kind: TokenKind) -> bool {
|
|
59
|
+
crate::core::parser::driver::cursor::match_token_impl(self, kind)
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
pub fn previous_clone(&self) -> Option<Token> {
|
|
63
|
+
crate::core::parser::driver::cursor::previous_clone_impl(self)
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
pub fn parse_block(
|
|
67
|
+
&self,
|
|
68
|
+
tokens: Vec<Token>,
|
|
69
|
+
global_store: &mut GlobalStore,
|
|
70
|
+
) -> Vec<Statement> {
|
|
71
|
+
crate::core::parser::driver::block::parse_block(self, tokens, global_store)
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
pub fn parse_tokens(
|
|
75
|
+
&mut self,
|
|
76
|
+
tokens: Vec<Token>,
|
|
77
|
+
global_store: &mut GlobalStore,
|
|
78
|
+
) -> Vec<Statement> {
|
|
79
|
+
crate::core::parser::driver::driver_impl::parse_tokens_impl(self, tokens, global_store)
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
pub fn check_token(&self, kind: TokenKind) -> bool {
|
|
83
|
+
crate::core::parser::driver::cursor::peek_impl(self).is_some_and(|t| t.kind == kind)
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
pub fn peek_kind(&self) -> Option<TokenKind> {
|
|
87
|
+
crate::core::parser::driver::cursor::peek_impl(self).map(|t| t.kind.clone())
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
pub fn parse_map_value(&mut self) -> Option<Value> {
|
|
91
|
+
// Delegated to parse_map.rs
|
|
92
|
+
crate::core::parser::driver::parse_map::parse_map_value(self)
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
// Parse an array value like [1, 2, 3] or ["a", b]
|
|
96
|
+
pub fn parse_array_value(&mut self) -> Option<Value> {
|
|
97
|
+
// delegated to parse_array.rs
|
|
98
|
+
crate::core::parser::driver::parse_array::parse_array_value(self)
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
pub fn peek(&self) -> Option<&Token> {
|
|
102
|
+
self.tokens.get(self.token_index)
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
pub fn peek_clone(&self) -> Option<Token> {
|
|
106
|
+
self.tokens.get(self.token_index).cloned()
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
pub fn expect(&mut self, kind: TokenKind) -> Result<&Token, String> {
|
|
110
|
+
let tok = self.advance().ok_or("Unexpected end of input")?;
|
|
111
|
+
if tok.kind == kind {
|
|
112
|
+
Ok(tok)
|
|
113
|
+
} else {
|
|
114
|
+
Err(format!("Expected {:?}, got {:?}", kind, tok.kind))
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
|
|
118
|
+
pub fn collect_block_tokens(&mut self, base_indent: usize) -> Vec<Token> {
|
|
119
|
+
crate::core::parser::driver::block::collect_block_tokens(self, base_indent)
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
pub fn collect_until<F>(&mut self, condition: F) -> Vec<Token>
|
|
123
|
+
where
|
|
124
|
+
F: Fn(&Token) -> bool,
|
|
125
|
+
{
|
|
126
|
+
crate::core::parser::driver::block::collect_until(self, condition)
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
pub fn is_eof(&self) -> bool {
|
|
130
|
+
self.token_index >= self.tokens.len()
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
pub fn parse_block_until_next_else(
|
|
134
|
+
&mut self,
|
|
135
|
+
base_indent: usize,
|
|
136
|
+
global_store: &mut GlobalStore,
|
|
137
|
+
) -> Vec<Statement> {
|
|
138
|
+
crate::core::parser::driver::block::parse_block_until_next_else(
|
|
139
|
+
self,
|
|
140
|
+
base_indent,
|
|
141
|
+
global_store,
|
|
142
|
+
)
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
pub fn parse_condition_until_colon(&mut self) -> Option<Value> {
|
|
146
|
+
crate::core::parser::driver::driver_impl::parse_condition_until_colon_impl(self)
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
pub fn parse_block_until_else_or_dedent(
|
|
150
|
+
&mut self,
|
|
151
|
+
base_indent: usize,
|
|
152
|
+
global_store: &mut GlobalStore,
|
|
153
|
+
) -> Vec<Statement> {
|
|
154
|
+
crate::core::parser::driver::block::parse_block_until_else_or_dedent(
|
|
155
|
+
self,
|
|
156
|
+
base_indent,
|
|
157
|
+
global_store,
|
|
158
|
+
)
|
|
159
|
+
}
|
|
160
|
+
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
use crate::core::{
|
|
2
2
|
lexer::token::TokenKind,
|
|
3
3
|
parser::{
|
|
4
|
-
driver::Parser,
|
|
4
|
+
driver::parser::Parser,
|
|
5
5
|
statement::{Statement, StatementKind},
|
|
6
6
|
},
|
|
7
7
|
store::global::GlobalStore,
|
|
@@ -66,10 +66,15 @@ fn parse_map_literal(parser: &mut Parser) -> Value {
|
|
|
66
66
|
if let Some(TokenKind::Slash) = parser.peek_kind() {
|
|
67
67
|
parser.advance(); // '/'
|
|
68
68
|
if let Some(den) = parser.peek_clone() {
|
|
69
|
-
if den.kind == TokenKind::Number
|
|
69
|
+
if den.kind == TokenKind::Number
|
|
70
|
+
|| den.kind == TokenKind::Identifier
|
|
71
|
+
{
|
|
70
72
|
parser.advance();
|
|
71
73
|
let beat = format!("{}/{}", value_token.lexeme, den.lexeme);
|
|
72
|
-
map.insert(
|
|
74
|
+
map.insert(
|
|
75
|
+
key,
|
|
76
|
+
Value::Duration(devalang_types::Duration::Beat(beat)),
|
|
77
|
+
);
|
|
73
78
|
continue;
|
|
74
79
|
}
|
|
75
80
|
}
|
|
@@ -219,7 +224,26 @@ pub fn parse_arrow_call(parser: &mut Parser, _global_store: &mut GlobalStore) ->
|
|
|
219
224
|
let value = match token.kind {
|
|
220
225
|
TokenKind::Identifier => Value::Identifier(token.lexeme.clone()),
|
|
221
226
|
TokenKind::String => Value::String(token.lexeme.clone()),
|
|
222
|
-
TokenKind::Number =>
|
|
227
|
+
TokenKind::Number => {
|
|
228
|
+
// support fraction literal as bare arg: 1/4
|
|
229
|
+
// note: token has already been advanced earlier in the loop
|
|
230
|
+
if let Some(TokenKind::Slash) = parser.peek_kind() {
|
|
231
|
+
parser.advance(); // consume '/'
|
|
232
|
+
if let Some(den) = parser.peek_clone() {
|
|
233
|
+
if den.kind == TokenKind::Number || den.kind == TokenKind::Identifier {
|
|
234
|
+
parser.advance();
|
|
235
|
+
let beat = format!("{}/{}", token.lexeme, den.lexeme);
|
|
236
|
+
Value::Duration(devalang_types::Duration::Beat(beat))
|
|
237
|
+
} else {
|
|
238
|
+
Value::Number(token.lexeme.parse::<f32>().unwrap_or(0.0))
|
|
239
|
+
}
|
|
240
|
+
} else {
|
|
241
|
+
Value::Number(token.lexeme.parse::<f32>().unwrap_or(0.0))
|
|
242
|
+
}
|
|
243
|
+
} else {
|
|
244
|
+
Value::Number(token.lexeme.parse::<f32>().unwrap_or(0.0))
|
|
245
|
+
}
|
|
246
|
+
}
|
|
223
247
|
TokenKind::LBrace => {
|
|
224
248
|
// Handle map literal (supports nested maps)
|
|
225
249
|
|