@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
|
@@ -1,254 +1,254 @@
|
|
|
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
|
-
use std::collections::HashMap;
|
|
11
|
-
|
|
12
|
-
pub fn parse_automate_token(
|
|
13
|
-
parser: &mut Parser,
|
|
14
|
-
current_token: Token,
|
|
15
|
-
_global_store: &mut GlobalStore,
|
|
16
|
-
) -> Statement {
|
|
17
|
-
parser.advance(); // consume 'automate'
|
|
18
|
-
|
|
19
|
-
// Expect target identifier
|
|
20
|
-
let Some(target_token) = parser.peek_clone() else {
|
|
21
|
-
return crate::core::parser::statement::error_from_token(
|
|
22
|
-
current_token,
|
|
23
|
-
"Expected target after 'automate'".to_string(),
|
|
24
|
-
);
|
|
25
|
-
};
|
|
26
|
-
|
|
27
|
-
if target_token.kind != TokenKind::Identifier && target_token.kind != TokenKind::String {
|
|
28
|
-
return crate::core::parser::statement::error_from_token(
|
|
29
|
-
target_token,
|
|
30
|
-
"Expected valid target after 'automate'".to_string(),
|
|
31
|
-
);
|
|
32
|
-
}
|
|
33
|
-
parser.advance(); // consume target
|
|
34
|
-
|
|
35
|
-
// Expect ':'
|
|
36
|
-
let Some(colon_token) = parser.peek_clone() else {
|
|
37
|
-
return crate::core::parser::statement::error_from_token(
|
|
38
|
-
target_token,
|
|
39
|
-
"Expected ':' after automate target".to_string(),
|
|
40
|
-
);
|
|
41
|
-
};
|
|
42
|
-
if colon_token.kind != TokenKind::Colon {
|
|
43
|
-
return crate::core::parser::statement::error_from_token(
|
|
44
|
-
colon_token,
|
|
45
|
-
"Expected ':' after automate target".to_string(),
|
|
46
|
-
);
|
|
47
|
-
}
|
|
48
|
-
parser.advance(); // consume ':'
|
|
49
|
-
|
|
50
|
-
let base_indent = current_token.indent;
|
|
51
|
-
|
|
52
|
-
// Collect tokens inside block (indented > base_indent)
|
|
53
|
-
let mut index = parser.token_index;
|
|
54
|
-
let mut tokens_inside = Vec::new();
|
|
55
|
-
while index < parser.tokens.len() {
|
|
56
|
-
let tok = parser.tokens[index].clone();
|
|
57
|
-
if tok.indent <= base_indent && tok.kind != TokenKind::Newline {
|
|
58
|
-
break;
|
|
59
|
-
}
|
|
60
|
-
tokens_inside.push(tok);
|
|
61
|
-
index += 1;
|
|
62
|
-
}
|
|
63
|
-
parser.token_index = index;
|
|
64
|
-
|
|
65
|
-
// Now parse block manually to capture 'param' entries without reusing general parser kinds
|
|
66
|
-
let mut local = Parser {
|
|
67
|
-
resolve_modules: parser.resolve_modules,
|
|
68
|
-
tokens: tokens_inside,
|
|
69
|
-
token_index: 0,
|
|
70
|
-
current_module: parser.current_module.clone(),
|
|
71
|
-
previous: None,
|
|
72
|
-
};
|
|
73
|
-
|
|
74
|
-
let mut params: HashMap<String, Value> = HashMap::new();
|
|
75
|
-
|
|
76
|
-
while let Some(tok) = local.peek_clone() {
|
|
77
|
-
match tok.kind {
|
|
78
|
-
TokenKind::Identifier if tok.lexeme == "param" => {
|
|
79
|
-
local.advance(); // consume 'param'
|
|
80
|
-
// param name
|
|
81
|
-
let Some(name_tok) = local.peek_clone() else {
|
|
82
|
-
return crate::core::parser::statement::error_from_token(
|
|
83
|
-
tok,
|
|
84
|
-
"Expected parameter name after 'param'".to_string(),
|
|
85
|
-
);
|
|
86
|
-
};
|
|
87
|
-
if name_tok.kind != TokenKind::Identifier && name_tok.kind != TokenKind::String {
|
|
88
|
-
return crate::core::parser::statement::error_from_token(
|
|
89
|
-
name_tok,
|
|
90
|
-
"Expected valid parameter name".to_string(),
|
|
91
|
-
);
|
|
92
|
-
}
|
|
93
|
-
local.advance(); // consume name
|
|
94
|
-
|
|
95
|
-
// Expect '{'
|
|
96
|
-
if !local.match_token(TokenKind::LBrace) {
|
|
97
|
-
return crate::core::parser::statement::error_from_token(
|
|
98
|
-
name_tok,
|
|
99
|
-
"Expected '{' to start parameter block".to_string(),
|
|
100
|
-
);
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
// Collect entries like: 0% = 0.0
|
|
104
|
-
let mut envelope: HashMap<String, Value> = HashMap::new();
|
|
105
|
-
while let Some(inner) = local.peek_clone() {
|
|
106
|
-
if inner.kind == TokenKind::RBrace {
|
|
107
|
-
local.advance();
|
|
108
|
-
break;
|
|
109
|
-
}
|
|
110
|
-
// Skip formatting tokens inside the param block
|
|
111
|
-
if matches!(
|
|
112
|
-
inner.kind,
|
|
113
|
-
TokenKind::Newline
|
|
114
|
-
| TokenKind::Indent
|
|
115
|
-
| TokenKind::Dedent
|
|
116
|
-
| TokenKind::Comma
|
|
117
|
-
) {
|
|
118
|
-
local.advance();
|
|
119
|
-
continue;
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
// Read percentage token: could be number followed by '%' as Dot or Identifier? '%' not defined.
|
|
123
|
-
// Our lexer has no Percent token, so accept either Number or Identifier containing e.g. '0%'.
|
|
124
|
-
let percent_token = inner.clone();
|
|
125
|
-
local.advance();
|
|
126
|
-
|
|
127
|
-
let percent_key = percent_token.lexeme.clone();
|
|
128
|
-
|
|
129
|
-
// Expect '='
|
|
130
|
-
// Skip any stray formatting between key and '='
|
|
131
|
-
while let Some(t) = local.peek_kind() {
|
|
132
|
-
if matches!(
|
|
133
|
-
t,
|
|
134
|
-
TokenKind::Indent | TokenKind::Dedent | TokenKind::Newline
|
|
135
|
-
) {
|
|
136
|
-
local.advance();
|
|
137
|
-
continue;
|
|
138
|
-
}
|
|
139
|
-
break;
|
|
140
|
-
}
|
|
141
|
-
if !local.match_token(TokenKind::Equals) {
|
|
142
|
-
return crate::core::parser::statement::error_from_token(
|
|
143
|
-
percent_token,
|
|
144
|
-
"Expected '=' in param entry".to_string(),
|
|
145
|
-
);
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
// Read value (number or identifier)
|
|
149
|
-
// Skip formatting before value
|
|
150
|
-
while let Some(t) = local.peek_kind() {
|
|
151
|
-
if matches!(
|
|
152
|
-
t,
|
|
153
|
-
TokenKind::Indent | TokenKind::Dedent | TokenKind::Newline
|
|
154
|
-
) {
|
|
155
|
-
local.advance();
|
|
156
|
-
continue;
|
|
157
|
-
}
|
|
158
|
-
break;
|
|
159
|
-
}
|
|
160
|
-
|
|
161
|
-
let value = if let Some(vtok) = local.peek_clone() {
|
|
162
|
-
match vtok.kind {
|
|
163
|
-
// Handle negative numbers where '-' is lexed as Arrow
|
|
164
|
-
TokenKind::Arrow => {
|
|
165
|
-
// Check if next token is a number
|
|
166
|
-
let mut num_str = String::from("-");
|
|
167
|
-
local.advance(); // consume '-'
|
|
168
|
-
if let Some(ntok) = local.peek_clone() {
|
|
169
|
-
if ntok.kind == TokenKind::Number {
|
|
170
|
-
num_str.push_str(&ntok.lexeme);
|
|
171
|
-
local.advance(); // consume number
|
|
172
|
-
if let Some(dot) = local.peek_clone() {
|
|
173
|
-
if dot.kind == TokenKind::Dot {
|
|
174
|
-
local.advance();
|
|
175
|
-
if let Some(frac) = local.peek_clone() {
|
|
176
|
-
if frac.kind == TokenKind::Number {
|
|
177
|
-
num_str.push('.');
|
|
178
|
-
num_str.push_str(&frac.lexeme);
|
|
179
|
-
local.advance();
|
|
180
|
-
}
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
}
|
|
184
|
-
Value::Number(num_str.parse::<f32>().unwrap_or(0.0))
|
|
185
|
-
} else {
|
|
186
|
-
Value::Unknown
|
|
187
|
-
}
|
|
188
|
-
} else {
|
|
189
|
-
Value::Unknown
|
|
190
|
-
}
|
|
191
|
-
}
|
|
192
|
-
TokenKind::Number => {
|
|
193
|
-
// Possibly a float with dot
|
|
194
|
-
let mut number_str = vtok.lexeme.clone();
|
|
195
|
-
local.advance();
|
|
196
|
-
if let Some(dot) = local.peek_clone() {
|
|
197
|
-
if dot.kind == TokenKind::Dot {
|
|
198
|
-
local.advance();
|
|
199
|
-
if let Some(frac) = local.peek_clone() {
|
|
200
|
-
if frac.kind == TokenKind::Number {
|
|
201
|
-
number_str.push('.');
|
|
202
|
-
number_str.push_str(&frac.lexeme);
|
|
203
|
-
local.advance();
|
|
204
|
-
}
|
|
205
|
-
}
|
|
206
|
-
}
|
|
207
|
-
}
|
|
208
|
-
Value::Number(number_str.parse::<f32>().unwrap_or(0.0))
|
|
209
|
-
}
|
|
210
|
-
TokenKind::Identifier => {
|
|
211
|
-
local.advance();
|
|
212
|
-
Value::Identifier(vtok.lexeme.clone())
|
|
213
|
-
}
|
|
214
|
-
TokenKind::String => {
|
|
215
|
-
local.advance();
|
|
216
|
-
Value::String(vtok.lexeme.clone())
|
|
217
|
-
}
|
|
218
|
-
_ => {
|
|
219
|
-
local.advance();
|
|
220
|
-
Value::Unknown
|
|
221
|
-
}
|
|
222
|
-
}
|
|
223
|
-
} else {
|
|
224
|
-
Value::Null
|
|
225
|
-
};
|
|
226
|
-
|
|
227
|
-
envelope.insert(percent_key, value);
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
params.insert(name_tok.lexeme.clone(), Value::Map(envelope));
|
|
231
|
-
}
|
|
232
|
-
_ => {
|
|
233
|
-
local.advance();
|
|
234
|
-
}
|
|
235
|
-
}
|
|
236
|
-
}
|
|
237
|
-
|
|
238
|
-
let mut value_map = HashMap::new();
|
|
239
|
-
value_map.insert(
|
|
240
|
-
"target".to_string(),
|
|
241
|
-
Value::String(target_token.lexeme.clone()),
|
|
242
|
-
);
|
|
243
|
-
value_map.insert("params".to_string(), Value::Map(params));
|
|
244
|
-
|
|
245
|
-
Statement {
|
|
246
|
-
kind: StatementKind::Automate {
|
|
247
|
-
target: target_token.lexeme.clone(),
|
|
248
|
-
},
|
|
249
|
-
value: Value::Map(value_map),
|
|
250
|
-
indent: current_token.indent,
|
|
251
|
-
line: current_token.line,
|
|
252
|
-
column: current_token.column,
|
|
253
|
-
}
|
|
254
|
-
}
|
|
1
|
+
use crate::core::{
|
|
2
|
+
lexer::token::{Token, TokenKind},
|
|
3
|
+
parser::{
|
|
4
|
+
driver::parser::Parser,
|
|
5
|
+
statement::{Statement, StatementKind},
|
|
6
|
+
},
|
|
7
|
+
store::global::GlobalStore,
|
|
8
|
+
};
|
|
9
|
+
use devalang_types::Value;
|
|
10
|
+
use std::collections::HashMap;
|
|
11
|
+
|
|
12
|
+
pub fn parse_automate_token(
|
|
13
|
+
parser: &mut Parser,
|
|
14
|
+
current_token: Token,
|
|
15
|
+
_global_store: &mut GlobalStore,
|
|
16
|
+
) -> Statement {
|
|
17
|
+
parser.advance(); // consume 'automate'
|
|
18
|
+
|
|
19
|
+
// Expect target identifier
|
|
20
|
+
let Some(target_token) = parser.peek_clone() else {
|
|
21
|
+
return crate::core::parser::statement::error_from_token(
|
|
22
|
+
current_token,
|
|
23
|
+
"Expected target after 'automate'".to_string(),
|
|
24
|
+
);
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
if target_token.kind != TokenKind::Identifier && target_token.kind != TokenKind::String {
|
|
28
|
+
return crate::core::parser::statement::error_from_token(
|
|
29
|
+
target_token,
|
|
30
|
+
"Expected valid target after 'automate'".to_string(),
|
|
31
|
+
);
|
|
32
|
+
}
|
|
33
|
+
parser.advance(); // consume target
|
|
34
|
+
|
|
35
|
+
// Expect ':'
|
|
36
|
+
let Some(colon_token) = parser.peek_clone() else {
|
|
37
|
+
return crate::core::parser::statement::error_from_token(
|
|
38
|
+
target_token,
|
|
39
|
+
"Expected ':' after automate target".to_string(),
|
|
40
|
+
);
|
|
41
|
+
};
|
|
42
|
+
if colon_token.kind != TokenKind::Colon {
|
|
43
|
+
return crate::core::parser::statement::error_from_token(
|
|
44
|
+
colon_token,
|
|
45
|
+
"Expected ':' after automate target".to_string(),
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
parser.advance(); // consume ':'
|
|
49
|
+
|
|
50
|
+
let base_indent = current_token.indent;
|
|
51
|
+
|
|
52
|
+
// Collect tokens inside block (indented > base_indent)
|
|
53
|
+
let mut index = parser.token_index;
|
|
54
|
+
let mut tokens_inside = Vec::new();
|
|
55
|
+
while index < parser.tokens.len() {
|
|
56
|
+
let tok = parser.tokens[index].clone();
|
|
57
|
+
if tok.indent <= base_indent && tok.kind != TokenKind::Newline {
|
|
58
|
+
break;
|
|
59
|
+
}
|
|
60
|
+
tokens_inside.push(tok);
|
|
61
|
+
index += 1;
|
|
62
|
+
}
|
|
63
|
+
parser.token_index = index;
|
|
64
|
+
|
|
65
|
+
// Now parse block manually to capture 'param' entries without reusing general parser kinds
|
|
66
|
+
let mut local = Parser {
|
|
67
|
+
resolve_modules: parser.resolve_modules,
|
|
68
|
+
tokens: tokens_inside,
|
|
69
|
+
token_index: 0,
|
|
70
|
+
current_module: parser.current_module.clone(),
|
|
71
|
+
previous: None,
|
|
72
|
+
};
|
|
73
|
+
|
|
74
|
+
let mut params: HashMap<String, Value> = HashMap::new();
|
|
75
|
+
|
|
76
|
+
while let Some(tok) = local.peek_clone() {
|
|
77
|
+
match tok.kind {
|
|
78
|
+
TokenKind::Identifier if tok.lexeme == "param" => {
|
|
79
|
+
local.advance(); // consume 'param'
|
|
80
|
+
// param name
|
|
81
|
+
let Some(name_tok) = local.peek_clone() else {
|
|
82
|
+
return crate::core::parser::statement::error_from_token(
|
|
83
|
+
tok,
|
|
84
|
+
"Expected parameter name after 'param'".to_string(),
|
|
85
|
+
);
|
|
86
|
+
};
|
|
87
|
+
if name_tok.kind != TokenKind::Identifier && name_tok.kind != TokenKind::String {
|
|
88
|
+
return crate::core::parser::statement::error_from_token(
|
|
89
|
+
name_tok,
|
|
90
|
+
"Expected valid parameter name".to_string(),
|
|
91
|
+
);
|
|
92
|
+
}
|
|
93
|
+
local.advance(); // consume name
|
|
94
|
+
|
|
95
|
+
// Expect '{'
|
|
96
|
+
if !local.match_token(TokenKind::LBrace) {
|
|
97
|
+
return crate::core::parser::statement::error_from_token(
|
|
98
|
+
name_tok,
|
|
99
|
+
"Expected '{' to start parameter block".to_string(),
|
|
100
|
+
);
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
// Collect entries like: 0% = 0.0
|
|
104
|
+
let mut envelope: HashMap<String, Value> = HashMap::new();
|
|
105
|
+
while let Some(inner) = local.peek_clone() {
|
|
106
|
+
if inner.kind == TokenKind::RBrace {
|
|
107
|
+
local.advance();
|
|
108
|
+
break;
|
|
109
|
+
}
|
|
110
|
+
// Skip formatting tokens inside the param block
|
|
111
|
+
if matches!(
|
|
112
|
+
inner.kind,
|
|
113
|
+
TokenKind::Newline
|
|
114
|
+
| TokenKind::Indent
|
|
115
|
+
| TokenKind::Dedent
|
|
116
|
+
| TokenKind::Comma
|
|
117
|
+
) {
|
|
118
|
+
local.advance();
|
|
119
|
+
continue;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
// Read percentage token: could be number followed by '%' as Dot or Identifier? '%' not defined.
|
|
123
|
+
// Our lexer has no Percent token, so accept either Number or Identifier containing e.g. '0%'.
|
|
124
|
+
let percent_token = inner.clone();
|
|
125
|
+
local.advance();
|
|
126
|
+
|
|
127
|
+
let percent_key = percent_token.lexeme.clone();
|
|
128
|
+
|
|
129
|
+
// Expect '='
|
|
130
|
+
// Skip any stray formatting between key and '='
|
|
131
|
+
while let Some(t) = local.peek_kind() {
|
|
132
|
+
if matches!(
|
|
133
|
+
t,
|
|
134
|
+
TokenKind::Indent | TokenKind::Dedent | TokenKind::Newline
|
|
135
|
+
) {
|
|
136
|
+
local.advance();
|
|
137
|
+
continue;
|
|
138
|
+
}
|
|
139
|
+
break;
|
|
140
|
+
}
|
|
141
|
+
if !local.match_token(TokenKind::Equals) {
|
|
142
|
+
return crate::core::parser::statement::error_from_token(
|
|
143
|
+
percent_token,
|
|
144
|
+
"Expected '=' in param entry".to_string(),
|
|
145
|
+
);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
// Read value (number or identifier)
|
|
149
|
+
// Skip formatting before value
|
|
150
|
+
while let Some(t) = local.peek_kind() {
|
|
151
|
+
if matches!(
|
|
152
|
+
t,
|
|
153
|
+
TokenKind::Indent | TokenKind::Dedent | TokenKind::Newline
|
|
154
|
+
) {
|
|
155
|
+
local.advance();
|
|
156
|
+
continue;
|
|
157
|
+
}
|
|
158
|
+
break;
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
let value = if let Some(vtok) = local.peek_clone() {
|
|
162
|
+
match vtok.kind {
|
|
163
|
+
// Handle negative numbers where '-' is lexed as Arrow
|
|
164
|
+
TokenKind::Arrow => {
|
|
165
|
+
// Check if next token is a number
|
|
166
|
+
let mut num_str = String::from("-");
|
|
167
|
+
local.advance(); // consume '-'
|
|
168
|
+
if let Some(ntok) = local.peek_clone() {
|
|
169
|
+
if ntok.kind == TokenKind::Number {
|
|
170
|
+
num_str.push_str(&ntok.lexeme);
|
|
171
|
+
local.advance(); // consume number
|
|
172
|
+
if let Some(dot) = local.peek_clone() {
|
|
173
|
+
if dot.kind == TokenKind::Dot {
|
|
174
|
+
local.advance();
|
|
175
|
+
if let Some(frac) = local.peek_clone() {
|
|
176
|
+
if frac.kind == TokenKind::Number {
|
|
177
|
+
num_str.push('.');
|
|
178
|
+
num_str.push_str(&frac.lexeme);
|
|
179
|
+
local.advance();
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
Value::Number(num_str.parse::<f32>().unwrap_or(0.0))
|
|
185
|
+
} else {
|
|
186
|
+
Value::Unknown
|
|
187
|
+
}
|
|
188
|
+
} else {
|
|
189
|
+
Value::Unknown
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
TokenKind::Number => {
|
|
193
|
+
// Possibly a float with dot
|
|
194
|
+
let mut number_str = vtok.lexeme.clone();
|
|
195
|
+
local.advance();
|
|
196
|
+
if let Some(dot) = local.peek_clone() {
|
|
197
|
+
if dot.kind == TokenKind::Dot {
|
|
198
|
+
local.advance();
|
|
199
|
+
if let Some(frac) = local.peek_clone() {
|
|
200
|
+
if frac.kind == TokenKind::Number {
|
|
201
|
+
number_str.push('.');
|
|
202
|
+
number_str.push_str(&frac.lexeme);
|
|
203
|
+
local.advance();
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
Value::Number(number_str.parse::<f32>().unwrap_or(0.0))
|
|
209
|
+
}
|
|
210
|
+
TokenKind::Identifier => {
|
|
211
|
+
local.advance();
|
|
212
|
+
Value::Identifier(vtok.lexeme.clone())
|
|
213
|
+
}
|
|
214
|
+
TokenKind::String => {
|
|
215
|
+
local.advance();
|
|
216
|
+
Value::String(vtok.lexeme.clone())
|
|
217
|
+
}
|
|
218
|
+
_ => {
|
|
219
|
+
local.advance();
|
|
220
|
+
Value::Unknown
|
|
221
|
+
}
|
|
222
|
+
}
|
|
223
|
+
} else {
|
|
224
|
+
Value::Null
|
|
225
|
+
};
|
|
226
|
+
|
|
227
|
+
envelope.insert(percent_key, value);
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
params.insert(name_tok.lexeme.clone(), Value::Map(envelope));
|
|
231
|
+
}
|
|
232
|
+
_ => {
|
|
233
|
+
local.advance();
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
|
|
238
|
+
let mut value_map = HashMap::new();
|
|
239
|
+
value_map.insert(
|
|
240
|
+
"target".to_string(),
|
|
241
|
+
Value::String(target_token.lexeme.clone()),
|
|
242
|
+
);
|
|
243
|
+
value_map.insert("params".to_string(), Value::Map(params));
|
|
244
|
+
|
|
245
|
+
Statement {
|
|
246
|
+
kind: StatementKind::Automate {
|
|
247
|
+
target: target_token.lexeme.clone(),
|
|
248
|
+
},
|
|
249
|
+
value: Value::Map(value_map),
|
|
250
|
+
indent: current_token.indent,
|
|
251
|
+
line: current_token.line,
|
|
252
|
+
column: current_token.column,
|
|
253
|
+
}
|
|
254
|
+
}
|