@devaloop/devalang 0.0.1-beta.3 → 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +182 -152
- package/out-tsc/api.d.ts +180 -0
- package/out-tsc/api.d.ts.map +1 -0
- package/out-tsc/api.js +286 -0
- package/out-tsc/api.js.map +1 -0
- package/out-tsc/bin/index.d.ts +12 -0
- package/out-tsc/bin/index.d.ts.map +1 -0
- package/out-tsc/bin/index.js +20 -54
- package/out-tsc/bin/index.js.map +1 -0
- package/out-tsc/examples/basic-usage.d.ts +8 -0
- package/out-tsc/examples/basic-usage.d.ts.map +1 -0
- package/out-tsc/examples/basic-usage.js +113 -0
- package/out-tsc/examples/basic-usage.js.map +1 -0
- package/out-tsc/index.d.ts +19 -5
- package/out-tsc/index.d.ts.map +1 -0
- package/out-tsc/index.js +24 -6
- package/out-tsc/index.js.map +1 -0
- package/out-tsc/scripts/copy-wasm-dts.d.ts +7 -0
- package/out-tsc/scripts/copy-wasm-dts.d.ts.map +1 -0
- package/out-tsc/scripts/copy-wasm-dts.js +36 -32
- package/out-tsc/scripts/copy-wasm-dts.js.map +1 -0
- package/out-tsc/scripts/postinstall.d.ts +1 -0
- package/out-tsc/scripts/postinstall.d.ts.map +1 -0
- package/out-tsc/scripts/postinstall.js +4 -1
- package/out-tsc/scripts/postinstall.js.map +1 -0
- package/out-tsc/scripts/version/bump.d.ts +5 -1
- package/out-tsc/scripts/version/bump.d.ts.map +1 -0
- package/out-tsc/scripts/version/bump.js +114 -44
- package/out-tsc/scripts/version/bump.js.map +1 -0
- package/out-tsc/scripts/version/fetch.d.ts +12 -1
- package/out-tsc/scripts/version/fetch.d.ts.map +1 -0
- package/out-tsc/scripts/version/fetch.js +68 -24
- package/out-tsc/scripts/version/fetch.js.map +1 -0
- package/out-tsc/scripts/version/index.d.ts +6 -0
- package/out-tsc/scripts/version/index.d.ts.map +1 -0
- package/out-tsc/scripts/version/index.js +44 -22
- package/out-tsc/scripts/version/index.js.map +1 -0
- package/out-tsc/scripts/version/sync.d.ts +5 -1
- package/out-tsc/scripts/version/sync.d.ts.map +1 -0
- package/out-tsc/scripts/version/sync.js +78 -29
- package/out-tsc/scripts/version/sync.js.map +1 -0
- package/out-tsc/types.d.ts +68 -0
- package/out-tsc/types.d.ts.map +1 -0
- package/out-tsc/{core/types/value.js → types.js} +4 -0
- package/out-tsc/types.js.map +1 -0
- package/out-tsc/wasm.d.ts +8 -0
- package/out-tsc/wasm.d.ts.map +1 -0
- package/out-tsc/{core/index.js → wasm.js} +9 -6
- package/out-tsc/wasm.js.map +1 -0
- package/package.json +42 -42
- package/.cargo/config.toml +0 -2
- package/.devalang +0 -9
- package/.github/workflows/ci.yml +0 -103
- package/Cargo.toml +0 -84
- package/docs/CHANGELOG.md +0 -622
- package/docs/CONTRIBUTING.md +0 -101
- package/docs/ROADMAP.md +0 -38
- package/docs/TODO.md +0 -71
- package/examples/automation.deva +0 -42
- package/examples/bank.deva +0 -7
- package/examples/bus.deva +0 -10
- package/examples/chain.deva +0 -19
- package/examples/condition.deva +0 -20
- package/examples/duration.deva +0 -9
- package/examples/effect.deva +0 -2
- package/examples/events.deva +0 -12
- package/examples/filter.deva +0 -11
- package/examples/function.deva +0 -15
- package/examples/group.deva +0 -12
- package/examples/index.deva +0 -63
- package/examples/lfo.deva +0 -9
- package/examples/loop.deva +0 -10
- package/examples/pattern.deva +0 -8
- package/examples/plugin.deva +0 -16
- package/examples/routing.deva +0 -23
- package/examples/samples/hat-808.wav +0 -0
- package/examples/samples/kick-808.wav +0 -0
- package/examples/synth.deva +0 -24
- package/examples/synth_types.deva +0 -17
- package/examples/variables.deva +0 -9
- package/out-tsc/bin/project-version.json +0 -6
- package/out-tsc/core/functions/index.d.ts +0 -42
- package/out-tsc/core/functions/index.js +0 -87
- package/out-tsc/core/index.d.ts +0 -6
- package/out-tsc/core/types/index.d.ts +0 -4
- package/out-tsc/core/types/index.js +0 -20
- package/out-tsc/core/types/plugin.d.ts +0 -18
- package/out-tsc/core/types/plugin.js +0 -2
- package/out-tsc/core/types/result.d.ts +0 -27
- package/out-tsc/core/types/result.js +0 -2
- package/out-tsc/core/types/statement.d.ts +0 -106
- package/out-tsc/core/types/statement.js +0 -2
- package/out-tsc/core/types/value.d.ts +0 -43
- package/out-tsc/pkg/devalang_core.d.ts +0 -15
- package/out-tsc/pkg/devalang_core.js +0 -65
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +0 -34
- package/out-tsc/scripts/version/copy-to-binary.d.ts +0 -1
- package/out-tsc/scripts/version/copy-to-binary.js +0 -79
- package/project-version.json +0 -6
- package/rust/bindings/Cargo.toml +0 -9
- package/rust/bindings/src/lib.rs +0 -86
- package/rust/cli/addon/commands.rs +0 -35
- package/rust/cli/addon/download.rs +0 -234
- package/rust/cli/addon/install.rs +0 -33
- package/rust/cli/addon/list.rs +0 -224
- package/rust/cli/addon/metadata.rs +0 -124
- package/rust/cli/addon/mod.rs +0 -8
- package/rust/cli/addon/remove.rs +0 -271
- package/rust/cli/addon/update.rs +0 -305
- package/rust/cli/addon/utils.rs +0 -109
- package/rust/cli/build/commands.rs +0 -153
- package/rust/cli/build/mod.rs +0 -2
- package/rust/cli/build/process.rs +0 -165
- package/rust/cli/check/mod.rs +0 -208
- package/rust/cli/discover/commands.rs +0 -275
- package/rust/cli/discover/config.rs +0 -109
- package/rust/cli/discover/fs.rs +0 -19
- package/rust/cli/discover/install.rs +0 -214
- package/rust/cli/discover/metadata.rs +0 -48
- package/rust/cli/discover/mod.rs +0 -5
- package/rust/cli/init/commands.rs +0 -88
- package/rust/cli/init/mod.rs +0 -1
- package/rust/cli/login/commands.rs +0 -124
- package/rust/cli/login/mod.rs +0 -1
- package/rust/cli/me/commands.rs +0 -52
- package/rust/cli/me/mod.rs +0 -1
- package/rust/cli/mod.rs +0 -12
- package/rust/cli/parser.rs +0 -320
- package/rust/cli/play/commands.rs +0 -375
- package/rust/cli/play/io.rs +0 -17
- package/rust/cli/play/mod.rs +0 -5
- package/rust/cli/play/process.rs +0 -159
- package/rust/cli/play/realtime.rs +0 -91
- package/rust/cli/play/utils.rs +0 -23
- package/rust/cli/telemetry/commands.rs +0 -22
- package/rust/cli/telemetry/event_creator.rs +0 -80
- package/rust/cli/telemetry/mod.rs +0 -3
- package/rust/cli/telemetry/send.rs +0 -51
- package/rust/cli/template/commands.rs +0 -69
- package/rust/cli/template/mod.rs +0 -1
- package/rust/cli/update/commands.rs +0 -6
- package/rust/cli/update/mod.rs +0 -1
- package/rust/config/driver.rs +0 -112
- package/rust/config/mod.rs +0 -3
- package/rust/config/ops.rs +0 -26
- package/rust/config/settings.rs +0 -101
- package/rust/core/audio/engine/driver.rs +0 -237
- package/rust/core/audio/engine/export.rs +0 -169
- package/rust/core/audio/engine/helpers.rs +0 -178
- package/rust/core/audio/engine/mod.rs +0 -56
- package/rust/core/audio/engine/notes/dsp.rs +0 -88
- package/rust/core/audio/engine/notes/mod.rs +0 -53
- package/rust/core/audio/engine/notes/params.rs +0 -294
- package/rust/core/audio/engine/sample/insert.rs +0 -300
- package/rust/core/audio/engine/sample/mod.rs +0 -40
- package/rust/core/audio/engine/sample/padding.rs +0 -170
- package/rust/core/audio/evaluator/condition.rs +0 -61
- package/rust/core/audio/evaluator/mod.rs +0 -9
- package/rust/core/audio/evaluator/numeric.rs +0 -152
- package/rust/core/audio/evaluator/rhs.rs +0 -16
- package/rust/core/audio/evaluator/string_expr.rs +0 -94
- package/rust/core/audio/interpreter/driver.rs +0 -574
- package/rust/core/audio/interpreter/mod.rs +0 -2
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +0 -179
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +0 -398
- package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +0 -323
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +0 -3
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +0 -371
- package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +0 -3
- package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +0 -192
- package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +0 -24
- package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +0 -116
- package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +0 -97
- package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +0 -100
- package/rust/core/audio/interpreter/statements/automate.rs +0 -16
- package/rust/core/audio/interpreter/statements/call.rs +0 -325
- package/rust/core/audio/interpreter/statements/condition.rs +0 -72
- package/rust/core/audio/interpreter/statements/function.rs +0 -24
- package/rust/core/audio/interpreter/statements/let_.rs +0 -36
- package/rust/core/audio/interpreter/statements/load.rs +0 -17
- package/rust/core/audio/interpreter/statements/loop_.rs +0 -115
- package/rust/core/audio/interpreter/statements/mod.rs +0 -12
- package/rust/core/audio/interpreter/statements/sleep.rs +0 -28
- package/rust/core/audio/interpreter/statements/spawn.rs +0 -302
- package/rust/core/audio/interpreter/statements/tempo.rs +0 -40
- package/rust/core/audio/interpreter/statements/trigger.rs +0 -242
- package/rust/core/audio/loader/mod.rs +0 -1
- package/rust/core/audio/loader/trigger.rs +0 -98
- package/rust/core/audio/mod.rs +0 -6
- package/rust/core/audio/player.rs +0 -70
- package/rust/core/audio/special/easing.rs +0 -189
- package/rust/core/audio/special/env.rs +0 -45
- package/rust/core/audio/special/math.rs +0 -134
- package/rust/core/audio/special/mod.rs +0 -9
- package/rust/core/audio/special/modulator.rs +0 -143
- package/rust/core/builder/mod.rs +0 -129
- package/rust/core/debugger/lexer.rs +0 -27
- package/rust/core/debugger/logs.rs +0 -52
- package/rust/core/debugger/mod.rs +0 -30
- package/rust/core/debugger/preprocessor.rs +0 -27
- package/rust/core/debugger/store.rs +0 -38
- package/rust/core/error/mod.rs +0 -269
- package/rust/core/lexer/driver.rs +0 -59
- package/rust/core/lexer/handler/arrow.rs +0 -82
- package/rust/core/lexer/handler/at.rs +0 -21
- package/rust/core/lexer/handler/brace.rs +0 -41
- package/rust/core/lexer/handler/colon.rs +0 -21
- package/rust/core/lexer/handler/comment.rs +0 -30
- package/rust/core/lexer/handler/dot.rs +0 -21
- package/rust/core/lexer/handler/driver.rs +0 -337
- package/rust/core/lexer/handler/identifier.rs +0 -47
- package/rust/core/lexer/handler/indent.rs +0 -66
- package/rust/core/lexer/handler/mod.rs +0 -15
- package/rust/core/lexer/handler/newline.rs +0 -23
- package/rust/core/lexer/handler/number.rs +0 -31
- package/rust/core/lexer/handler/operator.rs +0 -46
- package/rust/core/lexer/handler/parenthesis.rs +0 -41
- package/rust/core/lexer/handler/slash.rs +0 -21
- package/rust/core/lexer/handler/string.rs +0 -63
- package/rust/core/lexer/mod.rs +0 -3
- package/rust/core/lexer/token.rs +0 -91
- package/rust/core/mod.rs +0 -9
- package/rust/core/parser/driver/block.rs +0 -111
- package/rust/core/parser/driver/cursor.rs +0 -82
- package/rust/core/parser/driver/driver_impl.rs +0 -159
- package/rust/core/parser/driver/mod.rs +0 -6
- package/rust/core/parser/driver/parse_array.rs +0 -120
- package/rust/core/parser/driver/parse_map.rs +0 -247
- package/rust/core/parser/driver/parser.rs +0 -160
- package/rust/core/parser/handler/arrow_call.rs +0 -328
- package/rust/core/parser/handler/at.rs +0 -279
- package/rust/core/parser/handler/bank.rs +0 -104
- package/rust/core/parser/handler/condition.rs +0 -83
- package/rust/core/parser/handler/dot.rs +0 -148
- package/rust/core/parser/handler/identifier/automate.rs +0 -254
- package/rust/core/parser/handler/identifier/call.rs +0 -91
- package/rust/core/parser/handler/identifier/emit.rs +0 -70
- package/rust/core/parser/handler/identifier/function.rs +0 -113
- package/rust/core/parser/handler/identifier/group.rs +0 -89
- package/rust/core/parser/handler/identifier/let_.rs +0 -173
- package/rust/core/parser/handler/identifier/mod.rs +0 -55
- package/rust/core/parser/handler/identifier/on.rs +0 -107
- package/rust/core/parser/handler/identifier/print.rs +0 -49
- package/rust/core/parser/handler/identifier/sleep.rs +0 -96
- package/rust/core/parser/handler/identifier/spawn.rs +0 -91
- package/rust/core/parser/handler/identifier/synth.rs +0 -171
- package/rust/core/parser/handler/loop_.rs +0 -194
- package/rust/core/parser/handler/mod.rs +0 -9
- package/rust/core/parser/handler/pattern.rs +0 -97
- package/rust/core/parser/handler/tempo.rs +0 -105
- package/rust/core/parser/mod.rs +0 -3
- package/rust/core/parser/statement.rs +0 -10
- package/rust/core/plugin/loader.rs +0 -137
- package/rust/core/plugin/mod.rs +0 -2
- package/rust/core/plugin/runner/mod.rs +0 -11
- package/rust/core/plugin/runner/non_wasm.rs +0 -481
- package/rust/core/plugin/runner/wasm32.rs +0 -44
- package/rust/core/preprocessor/loader/inject.rs +0 -313
- package/rust/core/preprocessor/loader/loader_helpers.rs +0 -110
- package/rust/core/preprocessor/loader/mod.rs +0 -235
- package/rust/core/preprocessor/mod.rs +0 -4
- package/rust/core/preprocessor/module.rs +0 -55
- package/rust/core/preprocessor/processor/handlers.rs +0 -107
- package/rust/core/preprocessor/processor/mod.rs +0 -1
- package/rust/core/preprocessor/resolver/bank.rs +0 -49
- package/rust/core/preprocessor/resolver/call.rs +0 -124
- package/rust/core/preprocessor/resolver/condition.rs +0 -95
- package/rust/core/preprocessor/resolver/driver.rs +0 -324
- package/rust/core/preprocessor/resolver/function.rs +0 -69
- package/rust/core/preprocessor/resolver/group.rs +0 -122
- package/rust/core/preprocessor/resolver/let_.rs +0 -32
- package/rust/core/preprocessor/resolver/loop_.rs +0 -318
- package/rust/core/preprocessor/resolver/mod.rs +0 -16
- package/rust/core/preprocessor/resolver/pattern.rs +0 -95
- package/rust/core/preprocessor/resolver/spawn.rs +0 -99
- package/rust/core/preprocessor/resolver/synth.rs +0 -54
- package/rust/core/preprocessor/resolver/tempo.rs +0 -48
- package/rust/core/preprocessor/resolver/trigger.rs +0 -116
- package/rust/core/preprocessor/resolver/value.rs +0 -176
- package/rust/core/store/global.rs +0 -57
- package/rust/core/store/mod.rs +0 -1
- package/rust/lib.rs +0 -323
- package/rust/macros/Cargo.toml +0 -14
- package/rust/macros/src/lib.rs +0 -52
- package/rust/main.rs +0 -557
- package/rust/types/Cargo.toml +0 -11
- package/rust/types/src/addons.rs +0 -57
- package/rust/types/src/ast.rs +0 -202
- package/rust/types/src/config.rs +0 -82
- package/rust/types/src/lib.rs +0 -15
- package/rust/types/src/plugin.rs +0 -20
- package/rust/types/src/store.rs +0 -139
- package/rust/types/src/telemetry.rs +0 -85
- package/rust/utils/Cargo.toml +0 -29
- package/rust/utils/src/error.rs +0 -186
- package/rust/utils/src/file.rs +0 -477
- package/rust/utils/src/first_usage.rs +0 -97
- package/rust/utils/src/lib.rs +0 -9
- package/rust/utils/src/logger.rs +0 -200
- package/rust/utils/src/path.rs +0 -158
- package/rust/utils/src/signature.rs +0 -41
- package/rust/utils/src/spinner.rs +0 -20
- package/rust/utils/src/version.rs +0 -58
- package/rust/utils/src/watcher.rs +0 -46
- package/rust/web/api.rs +0 -5
- package/rust/web/auth.rs +0 -5
- package/rust/web/cdn.rs +0 -34
- package/rust/web/forge.rs +0 -5
- package/rust/web/mod.rs +0 -5
- package/rust/web/sso.rs +0 -5
- package/templates/minimal/.devalang +0 -5
- package/templates/minimal/README.md +0 -218
- package/templates/minimal/src/index.deva +0 -2
- package/templates/welcome/.devalang +0 -5
- package/templates/welcome/README.md +0 -218
- package/templates/welcome/samples/kick-808.wav +0 -0
- package/templates/welcome/src/index.deva +0 -61
- package/templates/welcome/src/variables.deva +0 -3
- package/tests/integration.rs +0 -21
- package/tests/rust/cli_check_build.rs +0 -21
- package/tests/rust/cli_help.rs +0 -12
- package/tests/rust/cli_template_list.rs +0 -10
- package/tests/rust/cli_version.rs +0 -11
- package/tests/typescript/index.spec.ts +0 -136
- package/tests/typescript/playhead.spec.ts +0 -36
- package/tests/typescript/render_e2e.spec.ts +0 -77
- package/tsconfig.json +0 -115
- package/typescript/bin/index.ts +0 -28
- package/typescript/core/functions/index.ts +0 -94
- package/typescript/core/index.ts +0 -6
- package/typescript/core/types/index.ts +0 -4
- package/typescript/core/types/plugin.ts +0 -19
- package/typescript/core/types/result.ts +0 -29
- package/typescript/core/types/statement.ts +0 -47
- package/typescript/core/types/value.ts +0 -29
- package/typescript/index.ts +0 -8
- package/typescript/pkg/devalang_core.d.ts +0 -4
- package/typescript/pkg/devalang_core.ts +0 -65
- package/typescript/scripts/copy-wasm-dts.ts +0 -41
- package/typescript/scripts/postinstall.ts +0 -85
- package/typescript/scripts/version/bump.ts +0 -44
- package/typescript/scripts/version/copy-to-binary.ts +0 -82
- package/typescript/scripts/version/fetch.ts +0 -18
- package/typescript/scripts/version/index.ts +0 -25
- package/typescript/scripts/version/sync.ts +0 -24
|
@@ -1,96 +0,0 @@
|
|
|
1
|
-
use devalang_types::Value;
|
|
2
|
-
|
|
3
|
-
use crate::core::{
|
|
4
|
-
lexer::token::{Token, TokenKind},
|
|
5
|
-
parser::{
|
|
6
|
-
driver::parser::Parser,
|
|
7
|
-
statement::{Statement, StatementKind},
|
|
8
|
-
},
|
|
9
|
-
store::global::GlobalStore,
|
|
10
|
-
};
|
|
11
|
-
|
|
12
|
-
pub fn parse_sleep_token(
|
|
13
|
-
parser: &mut Parser,
|
|
14
|
-
current_token: Token,
|
|
15
|
-
_global_store: &mut GlobalStore,
|
|
16
|
-
) -> Statement {
|
|
17
|
-
parser.advance(); // consume "sleep"
|
|
18
|
-
|
|
19
|
-
// Accept number, decimal, fraction like 1/4 (-> Duration::Beat), string, or identifier
|
|
20
|
-
let duration_value = if let Some(token) = parser.peek_clone() {
|
|
21
|
-
match token.kind {
|
|
22
|
-
TokenKind::Number => {
|
|
23
|
-
let mut num = token.lexeme.clone();
|
|
24
|
-
parser.advance();
|
|
25
|
-
// decimal part
|
|
26
|
-
if let Some(dot) = parser.peek_clone() {
|
|
27
|
-
if dot.kind == TokenKind::Dot {
|
|
28
|
-
if let Some(next) = parser.peek_nth(1).cloned() {
|
|
29
|
-
if next.kind == TokenKind::Number {
|
|
30
|
-
parser.advance(); // consume dot
|
|
31
|
-
parser.advance(); // consume next number
|
|
32
|
-
num.push('.');
|
|
33
|
-
num.push_str(&next.lexeme);
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
// fraction form 1/4 -> Duration::Beat("1/4")
|
|
40
|
-
if let Some(slash) = parser.peek_clone() {
|
|
41
|
-
if slash.kind == TokenKind::Slash {
|
|
42
|
-
parser.advance();
|
|
43
|
-
if let Some(den) = parser.peek_clone() {
|
|
44
|
-
if den.kind == TokenKind::Number || den.kind == TokenKind::Identifier {
|
|
45
|
-
let frac = format!("{}/{}", num, den.lexeme);
|
|
46
|
-
parser.advance();
|
|
47
|
-
Value::Duration(devalang_types::Duration::Beat(frac))
|
|
48
|
-
} else {
|
|
49
|
-
return crate::core::parser::statement::error_from_token(
|
|
50
|
-
slash,
|
|
51
|
-
"Expected denominator after '/' in sleep".to_string(),
|
|
52
|
-
);
|
|
53
|
-
}
|
|
54
|
-
} else {
|
|
55
|
-
return crate::core::parser::statement::error_from_token(
|
|
56
|
-
slash,
|
|
57
|
-
"Expected denominator after '/' in sleep".to_string(),
|
|
58
|
-
);
|
|
59
|
-
}
|
|
60
|
-
} else {
|
|
61
|
-
Value::Number(num.parse().unwrap_or(0.0))
|
|
62
|
-
}
|
|
63
|
-
} else {
|
|
64
|
-
Value::Number(num.parse().unwrap_or(0.0))
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
TokenKind::String => {
|
|
68
|
-
parser.advance();
|
|
69
|
-
Value::String(token.lexeme.clone())
|
|
70
|
-
}
|
|
71
|
-
TokenKind::Identifier => {
|
|
72
|
-
parser.advance();
|
|
73
|
-
Value::Identifier(token.lexeme.clone())
|
|
74
|
-
}
|
|
75
|
-
_ => {
|
|
76
|
-
return crate::core::parser::statement::error_from_token(
|
|
77
|
-
token,
|
|
78
|
-
"Expected duration after 'sleep'".to_string(),
|
|
79
|
-
);
|
|
80
|
-
}
|
|
81
|
-
}
|
|
82
|
-
} else {
|
|
83
|
-
return crate::core::parser::statement::error_from_token(
|
|
84
|
-
current_token,
|
|
85
|
-
"Expected duration after 'sleep'".to_string(),
|
|
86
|
-
);
|
|
87
|
-
};
|
|
88
|
-
|
|
89
|
-
Statement {
|
|
90
|
-
kind: StatementKind::Sleep,
|
|
91
|
-
value: duration_value,
|
|
92
|
-
indent: current_token.indent,
|
|
93
|
-
line: current_token.line,
|
|
94
|
-
column: current_token.column,
|
|
95
|
-
}
|
|
96
|
-
}
|
|
@@ -1,91 +0,0 @@
|
|
|
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
|
-
|
|
11
|
-
pub fn parse_spawn_token(
|
|
12
|
-
parser: &mut Parser,
|
|
13
|
-
current_token: Token,
|
|
14
|
-
_global_store: &mut GlobalStore,
|
|
15
|
-
) -> Statement {
|
|
16
|
-
parser.advance(); // consume "spawn"
|
|
17
|
-
|
|
18
|
-
// Expect function name
|
|
19
|
-
let name_token = match parser.peek_clone() {
|
|
20
|
-
Some(t) => t,
|
|
21
|
-
None => {
|
|
22
|
-
return crate::core::parser::statement::error_from_token(
|
|
23
|
-
current_token,
|
|
24
|
-
"Expected function name after 'spawn'".to_string(),
|
|
25
|
-
);
|
|
26
|
-
}
|
|
27
|
-
};
|
|
28
|
-
|
|
29
|
-
if name_token.kind != TokenKind::Identifier {
|
|
30
|
-
return crate::core::parser::statement::error_from_token(
|
|
31
|
-
name_token,
|
|
32
|
-
"Expected function name to be an identifier".to_string(),
|
|
33
|
-
);
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
let func_name = name_token.lexeme.clone();
|
|
37
|
-
parser.advance(); // consume function name
|
|
38
|
-
|
|
39
|
-
// Expect '('
|
|
40
|
-
let mut args: Vec<Value> = Vec::new();
|
|
41
|
-
if let Some(open_paren) = parser.peek_clone() {
|
|
42
|
-
if open_paren.kind == TokenKind::LParen {
|
|
43
|
-
parser.advance(); // consume '('
|
|
44
|
-
|
|
45
|
-
// Collect args until ')'
|
|
46
|
-
while let Some(token) = parser.peek_clone() {
|
|
47
|
-
if token.kind == TokenKind::RParen {
|
|
48
|
-
parser.advance(); // consume ')'
|
|
49
|
-
break;
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
match token.kind {
|
|
53
|
-
TokenKind::Number => {
|
|
54
|
-
if let Ok(num) = token.lexeme.parse::<f32>() {
|
|
55
|
-
args.push(Value::Number(num));
|
|
56
|
-
}
|
|
57
|
-
parser.advance();
|
|
58
|
-
}
|
|
59
|
-
TokenKind::String => {
|
|
60
|
-
args.push(Value::String(token.lexeme.clone()));
|
|
61
|
-
parser.advance();
|
|
62
|
-
}
|
|
63
|
-
TokenKind::Identifier => {
|
|
64
|
-
args.push(Value::Identifier(token.lexeme.clone()));
|
|
65
|
-
parser.advance();
|
|
66
|
-
}
|
|
67
|
-
TokenKind::Comma => {
|
|
68
|
-
parser.advance(); // skip comma
|
|
69
|
-
}
|
|
70
|
-
_ => {
|
|
71
|
-
return crate::core::parser::statement::error_from_token(
|
|
72
|
-
token,
|
|
73
|
-
"Unexpected token in spawn arguments".to_string(),
|
|
74
|
-
);
|
|
75
|
-
}
|
|
76
|
-
}
|
|
77
|
-
}
|
|
78
|
-
}
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
Statement {
|
|
82
|
-
kind: StatementKind::Spawn {
|
|
83
|
-
name: func_name,
|
|
84
|
-
args,
|
|
85
|
-
},
|
|
86
|
-
value: Value::Null,
|
|
87
|
-
indent: current_token.indent,
|
|
88
|
-
line: current_token.line,
|
|
89
|
-
column: current_token.column,
|
|
90
|
-
}
|
|
91
|
-
}
|
|
@@ -1,171 +0,0 @@
|
|
|
1
|
-
use std::collections::HashMap;
|
|
2
|
-
|
|
3
|
-
use devalang_types::Value;
|
|
4
|
-
|
|
5
|
-
use crate::core::{
|
|
6
|
-
lexer::token::Token,
|
|
7
|
-
parser::{
|
|
8
|
-
driver::parser::Parser,
|
|
9
|
-
handler::dot::parse_dot_token,
|
|
10
|
-
statement::{Statement, StatementKind},
|
|
11
|
-
},
|
|
12
|
-
store::global::GlobalStore,
|
|
13
|
-
};
|
|
14
|
-
|
|
15
|
-
pub fn parse_synth_token(
|
|
16
|
-
parser: &mut Parser,
|
|
17
|
-
_current_token: Token,
|
|
18
|
-
_global_store: &mut GlobalStore,
|
|
19
|
-
) -> Statement {
|
|
20
|
-
parser.advance(); // consume 'synth'
|
|
21
|
-
|
|
22
|
-
let Some(synth_token) = parser.previous_clone() else {
|
|
23
|
-
return Statement::unknown();
|
|
24
|
-
};
|
|
25
|
-
|
|
26
|
-
// Expect a provider or waveform identifier (can be dotted: alias.synth)
|
|
27
|
-
// Also accept a dot-led entity by delegating to the dot parser (e.g. .module.export)
|
|
28
|
-
// Support optional explicit waveform after a dotted provider: `synth alias.synth saw {}`
|
|
29
|
-
let synth_waveform = if let Some(first_token) = parser.peek_clone() {
|
|
30
|
-
use crate::core::lexer::token::TokenKind;
|
|
31
|
-
|
|
32
|
-
if first_token.kind == TokenKind::Dot {
|
|
33
|
-
// Parse dot-entity and extract its entity string
|
|
34
|
-
let dot_stmt = parse_dot_token(parser, _global_store);
|
|
35
|
-
// Extract entity if the parsed statement is a Trigger
|
|
36
|
-
match dot_stmt.kind {
|
|
37
|
-
StatementKind::Trigger { entity, .. } => entity,
|
|
38
|
-
_ => String::new(),
|
|
39
|
-
}
|
|
40
|
-
} else {
|
|
41
|
-
if first_token.kind != crate::core::lexer::token::TokenKind::Identifier
|
|
42
|
-
&& first_token.kind != crate::core::lexer::token::TokenKind::Number
|
|
43
|
-
&& first_token.kind != crate::core::lexer::token::TokenKind::Synth
|
|
44
|
-
{
|
|
45
|
-
return crate::core::parser::statement::error_from_token(
|
|
46
|
-
first_token.clone(),
|
|
47
|
-
"Expected identifier after 'synth'".to_string(),
|
|
48
|
-
);
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
// Collect dotted parts on the same line
|
|
52
|
-
let mut parts: Vec<String> = Vec::new();
|
|
53
|
-
let current_line = first_token.line;
|
|
54
|
-
loop {
|
|
55
|
-
let Some(tok) = parser.peek_clone() else {
|
|
56
|
-
break;
|
|
57
|
-
};
|
|
58
|
-
if tok.line != current_line {
|
|
59
|
-
break;
|
|
60
|
-
}
|
|
61
|
-
match tok.kind {
|
|
62
|
-
crate::core::lexer::token::TokenKind::Identifier
|
|
63
|
-
| crate::core::lexer::token::TokenKind::Number
|
|
64
|
-
| crate::core::lexer::token::TokenKind::Synth => {
|
|
65
|
-
parts.push(tok.lexeme.clone());
|
|
66
|
-
parser.advance();
|
|
67
|
-
// If next isn't a dot on same line, stop
|
|
68
|
-
if let Some(next) = parser.peek_clone() {
|
|
69
|
-
if !(next.line == current_line
|
|
70
|
-
&& next.kind == crate::core::lexer::token::TokenKind::Dot)
|
|
71
|
-
{
|
|
72
|
-
break;
|
|
73
|
-
}
|
|
74
|
-
} else {
|
|
75
|
-
break;
|
|
76
|
-
}
|
|
77
|
-
}
|
|
78
|
-
crate::core::lexer::token::TokenKind::Dot => {
|
|
79
|
-
parser.advance();
|
|
80
|
-
}
|
|
81
|
-
_ => break,
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
// The joined parts form either:
|
|
86
|
-
// - a simple waveform name (e.g. "sine")
|
|
87
|
-
// - a provider entity (e.g. "author.synth")
|
|
88
|
-
let first = parts.join(".");
|
|
89
|
-
|
|
90
|
-
// If this looks like a dotted provider, try to consume an optional explicit
|
|
91
|
-
// waveform identifier on the same line (e.g. `synth author.synth saw`). If none
|
|
92
|
-
// is provided, fall back to the original behaviour (use the joined string as
|
|
93
|
-
// the waveform identifier) to remain backward-compatible.
|
|
94
|
-
if first.contains('.') {
|
|
95
|
-
// peek next token on same line
|
|
96
|
-
if let Some(next_tok) = parser.peek_clone() {
|
|
97
|
-
if next_tok.line == current_line
|
|
98
|
-
&& matches!(
|
|
99
|
-
next_tok.kind,
|
|
100
|
-
crate::core::lexer::token::TokenKind::Identifier
|
|
101
|
-
| crate::core::lexer::token::TokenKind::Number
|
|
102
|
-
| crate::core::lexer::token::TokenKind::Synth
|
|
103
|
-
)
|
|
104
|
-
{
|
|
105
|
-
// consume waveform token
|
|
106
|
-
let waveform = next_tok.lexeme.clone();
|
|
107
|
-
parser.advance();
|
|
108
|
-
// store as "provider.waveform" style? keep only waveform for compatibility
|
|
109
|
-
waveform
|
|
110
|
-
} else {
|
|
111
|
-
// no explicit waveform -> use the provider string (old behaviour)
|
|
112
|
-
first
|
|
113
|
-
}
|
|
114
|
-
} else {
|
|
115
|
-
first
|
|
116
|
-
}
|
|
117
|
-
} else {
|
|
118
|
-
// simple waveform name
|
|
119
|
-
first
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
} else {
|
|
123
|
-
return crate::core::parser::statement::error_from_token(
|
|
124
|
-
synth_token,
|
|
125
|
-
"Expected identifier after 'synth'".to_string(),
|
|
126
|
-
);
|
|
127
|
-
};
|
|
128
|
-
|
|
129
|
-
// Skip formatting before optional parameters map
|
|
130
|
-
while parser.check_token(crate::core::lexer::token::TokenKind::Newline)
|
|
131
|
-
|| parser.check_token(crate::core::lexer::token::TokenKind::Indent)
|
|
132
|
-
|| parser.check_token(crate::core::lexer::token::TokenKind::Dedent)
|
|
133
|
-
|| parser.check_token(crate::core::lexer::token::TokenKind::Whitespace)
|
|
134
|
-
{
|
|
135
|
-
parser.advance();
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
// Expect synth optional parameters map
|
|
139
|
-
let parameters = if let Some(params) = parser.parse_map_value() {
|
|
140
|
-
// If parameters are provided, we expect a map
|
|
141
|
-
if let Value::Map(map) = params {
|
|
142
|
-
map
|
|
143
|
-
} else {
|
|
144
|
-
return crate::core::parser::statement::error_from_token(
|
|
145
|
-
synth_token,
|
|
146
|
-
"Expected a map for synth parameters".to_string(),
|
|
147
|
-
);
|
|
148
|
-
}
|
|
149
|
-
} else {
|
|
150
|
-
// If no parameters are provided, we can still create the statement with an empty map
|
|
151
|
-
HashMap::new()
|
|
152
|
-
};
|
|
153
|
-
|
|
154
|
-
Statement {
|
|
155
|
-
kind: StatementKind::Synth,
|
|
156
|
-
value: Value::Map(HashMap::from([
|
|
157
|
-
("entity".to_string(), Value::String("synth".to_string())),
|
|
158
|
-
(
|
|
159
|
-
"value".to_string(),
|
|
160
|
-
Value::Map(HashMap::from([
|
|
161
|
-
// Store waveform as identifier to allow resolution from variables/exports
|
|
162
|
-
("waveform".to_string(), Value::Identifier(synth_waveform)),
|
|
163
|
-
("parameters".to_string(), Value::Map(parameters)),
|
|
164
|
-
])),
|
|
165
|
-
),
|
|
166
|
-
])),
|
|
167
|
-
indent: synth_token.indent,
|
|
168
|
-
line: synth_token.line,
|
|
169
|
-
column: synth_token.column,
|
|
170
|
-
}
|
|
171
|
-
}
|
|
@@ -1,194 +0,0 @@
|
|
|
1
|
-
use devalang_types::Value;
|
|
2
|
-
|
|
3
|
-
use crate::core::{
|
|
4
|
-
lexer::token::TokenKind,
|
|
5
|
-
parser::{
|
|
6
|
-
driver::parser::Parser,
|
|
7
|
-
statement::{Statement, StatementKind},
|
|
8
|
-
},
|
|
9
|
-
store::global::GlobalStore,
|
|
10
|
-
};
|
|
11
|
-
|
|
12
|
-
pub fn parse_loop_token(parser: &mut Parser, global_store: &mut GlobalStore) -> Statement {
|
|
13
|
-
parser.advance(); // consume 'loop' or 'for' (aliased in lexer)
|
|
14
|
-
let Some(loop_token) = parser.previous_clone() else {
|
|
15
|
-
return Statement::unknown();
|
|
16
|
-
};
|
|
17
|
-
|
|
18
|
-
// Support two forms:
|
|
19
|
-
// 1) loop <count>:
|
|
20
|
-
// 2) for <ident> in [a,b,c]:
|
|
21
|
-
|
|
22
|
-
// Peek next to decide
|
|
23
|
-
let Some(next_token) = parser.peek_clone() else {
|
|
24
|
-
return Statement::error_with_pos(
|
|
25
|
-
loop_token.indent,
|
|
26
|
-
loop_token.line,
|
|
27
|
-
loop_token.column,
|
|
28
|
-
"Expected iterator after loop/for".to_string(),
|
|
29
|
-
);
|
|
30
|
-
};
|
|
31
|
-
|
|
32
|
-
// Try to detect 'for <ident> in [array]:' form
|
|
33
|
-
let mut foreach_ident: Option<String> = None;
|
|
34
|
-
if let TokenKind::Identifier = next_token.kind {
|
|
35
|
-
// Could be either count identifier (old form) or foreach variable
|
|
36
|
-
// Look ahead for 'in'
|
|
37
|
-
let name = next_token.lexeme.clone();
|
|
38
|
-
// don't consume yet; we'll branch
|
|
39
|
-
if let Some(t2) = parser.peek_nth(1) {
|
|
40
|
-
if t2.kind == TokenKind::Identifier && t2.lexeme == "in" {
|
|
41
|
-
// foreach form
|
|
42
|
-
foreach_ident = Some(name);
|
|
43
|
-
// consume ident and 'in'
|
|
44
|
-
parser.advance();
|
|
45
|
-
parser.advance();
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
if let Some(var_name) = foreach_ident {
|
|
51
|
-
// Expect [array] OR number OR string OR identifier after 'in'
|
|
52
|
-
let array_val = if let Some(tok) = parser.peek_clone() {
|
|
53
|
-
match tok.kind {
|
|
54
|
-
TokenKind::LBracket => {
|
|
55
|
-
if let Some(v) = parser.parse_array_value() {
|
|
56
|
-
v
|
|
57
|
-
} else {
|
|
58
|
-
Value::Array(vec![])
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
|
-
TokenKind::Number => {
|
|
62
|
-
parser.advance();
|
|
63
|
-
let n = tok.lexeme.parse::<f32>().unwrap_or(0.0);
|
|
64
|
-
Value::Number(n)
|
|
65
|
-
}
|
|
66
|
-
TokenKind::String => {
|
|
67
|
-
parser.advance();
|
|
68
|
-
Value::String(tok.lexeme.clone())
|
|
69
|
-
}
|
|
70
|
-
TokenKind::Identifier => {
|
|
71
|
-
parser.advance();
|
|
72
|
-
Value::Identifier(tok.lexeme.clone())
|
|
73
|
-
}
|
|
74
|
-
_ => {
|
|
75
|
-
return Statement::error_with_pos(
|
|
76
|
-
loop_token.indent,
|
|
77
|
-
loop_token.line,
|
|
78
|
-
loop_token.column,
|
|
79
|
-
"Expected array, number, string or identifier after 'in'".to_string(),
|
|
80
|
-
);
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
} else {
|
|
84
|
-
return Statement::error_with_pos(
|
|
85
|
-
loop_token.indent,
|
|
86
|
-
loop_token.line,
|
|
87
|
-
loop_token.column,
|
|
88
|
-
"Expected array, number, string or identifier after 'in'".to_string(),
|
|
89
|
-
);
|
|
90
|
-
};
|
|
91
|
-
|
|
92
|
-
// Expect ':'
|
|
93
|
-
if !parser.match_token(TokenKind::Colon) {
|
|
94
|
-
return Statement::error_with_pos(
|
|
95
|
-
loop_token.indent,
|
|
96
|
-
loop_token.line,
|
|
97
|
-
loop_token.column,
|
|
98
|
-
"Expected ':' after foreach header".to_string(),
|
|
99
|
-
);
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
let tokens =
|
|
103
|
-
parser.collect_until(|t| t.kind == TokenKind::Dedent || t.kind == TokenKind::EOF);
|
|
104
|
-
let loop_body = parser.parse_block(tokens.clone(), global_store);
|
|
105
|
-
if let Some(token) = parser.peek() {
|
|
106
|
-
if token.kind == TokenKind::Dedent {
|
|
107
|
-
parser.advance();
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
let mut value_map = std::collections::HashMap::new();
|
|
112
|
-
value_map.insert("foreach".to_string(), Value::Identifier(var_name));
|
|
113
|
-
value_map.insert("array".to_string(), array_val);
|
|
114
|
-
value_map.insert("body".to_string(), Value::Block(loop_body.clone()));
|
|
115
|
-
|
|
116
|
-
return Statement {
|
|
117
|
-
kind: StatementKind::Loop,
|
|
118
|
-
value: Value::Map(value_map),
|
|
119
|
-
indent: loop_token.indent,
|
|
120
|
-
line: loop_token.line,
|
|
121
|
-
column: loop_token.column,
|
|
122
|
-
};
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
// Fallback to legacy: loop <count>:
|
|
126
|
-
let Some(iterator_token) = parser.peek_clone() else {
|
|
127
|
-
return Statement::error_with_pos(
|
|
128
|
-
loop_token.indent,
|
|
129
|
-
loop_token.line,
|
|
130
|
-
loop_token.column,
|
|
131
|
-
"Expected number or identifier after 'loop'".to_string(),
|
|
132
|
-
);
|
|
133
|
-
};
|
|
134
|
-
|
|
135
|
-
let iterator_value = match iterator_token.kind {
|
|
136
|
-
TokenKind::Number => {
|
|
137
|
-
let val = iterator_token.lexeme.parse::<f32>().unwrap_or(1.0);
|
|
138
|
-
parser.advance();
|
|
139
|
-
Value::Number(val)
|
|
140
|
-
}
|
|
141
|
-
TokenKind::Identifier => {
|
|
142
|
-
let val = iterator_token.lexeme.clone();
|
|
143
|
-
parser.advance();
|
|
144
|
-
Value::Identifier(val)
|
|
145
|
-
}
|
|
146
|
-
TokenKind::String => {
|
|
147
|
-
// strings that are numeric (e.g. "10")
|
|
148
|
-
let s = iterator_token.lexeme.clone();
|
|
149
|
-
parser.advance();
|
|
150
|
-
Value::String(s)
|
|
151
|
-
}
|
|
152
|
-
_ => {
|
|
153
|
-
return Statement::error_with_pos(
|
|
154
|
-
iterator_token.clone().indent,
|
|
155
|
-
iterator_token.clone().line,
|
|
156
|
-
iterator_token.clone().column,
|
|
157
|
-
"Expected a number, string or identifier as loop count".to_string(),
|
|
158
|
-
);
|
|
159
|
-
}
|
|
160
|
-
};
|
|
161
|
-
|
|
162
|
-
if !parser.match_token(TokenKind::Colon) {
|
|
163
|
-
let message = format!(
|
|
164
|
-
"Expected ':' after loop count, got {:?}",
|
|
165
|
-
parser.peek_kind()
|
|
166
|
-
);
|
|
167
|
-
return Statement::error_with_pos(
|
|
168
|
-
loop_token.clone().indent,
|
|
169
|
-
loop_token.clone().line,
|
|
170
|
-
loop_token.clone().column,
|
|
171
|
-
message,
|
|
172
|
-
);
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
let tokens = parser.collect_until(|t| t.kind == TokenKind::Dedent || t.kind == TokenKind::EOF);
|
|
176
|
-
let loop_body = parser.parse_block(tokens.clone(), global_store);
|
|
177
|
-
if let Some(token) = parser.peek() {
|
|
178
|
-
if token.kind == TokenKind::Dedent {
|
|
179
|
-
parser.advance();
|
|
180
|
-
}
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
let mut value_map = std::collections::HashMap::new();
|
|
184
|
-
value_map.insert("iterator".to_string(), iterator_value);
|
|
185
|
-
value_map.insert("body".to_string(), Value::Block(loop_body.clone()));
|
|
186
|
-
|
|
187
|
-
Statement {
|
|
188
|
-
kind: StatementKind::Loop,
|
|
189
|
-
value: Value::Map(value_map),
|
|
190
|
-
indent: loop_token.indent,
|
|
191
|
-
line: loop_token.line,
|
|
192
|
-
column: loop_token.column,
|
|
193
|
-
}
|
|
194
|
-
}
|
|
@@ -1,97 +0,0 @@
|
|
|
1
|
-
use devalang_types::Value;
|
|
2
|
-
|
|
3
|
-
use crate::core::{
|
|
4
|
-
lexer::token::TokenKind,
|
|
5
|
-
parser::{
|
|
6
|
-
driver::parser::Parser,
|
|
7
|
-
statement::{Statement, StatementKind},
|
|
8
|
-
},
|
|
9
|
-
store::global::GlobalStore,
|
|
10
|
-
};
|
|
11
|
-
|
|
12
|
-
pub fn parse_pattern_token(parser: &mut Parser, _global_store: &mut GlobalStore) -> Statement {
|
|
13
|
-
// consume 'pattern'
|
|
14
|
-
parser.advance();
|
|
15
|
-
|
|
16
|
-
let Some(tok) = parser.previous_clone() else {
|
|
17
|
-
return Statement::unknown();
|
|
18
|
-
};
|
|
19
|
-
|
|
20
|
-
// Parse pattern name
|
|
21
|
-
let mut name = String::new();
|
|
22
|
-
if let Some(next) = parser.peek_clone() {
|
|
23
|
-
if next.kind == TokenKind::Identifier {
|
|
24
|
-
parser.advance();
|
|
25
|
-
name = next.lexeme.clone();
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
// optional 'with <target>' sequence
|
|
30
|
-
let mut target: Option<String> = None;
|
|
31
|
-
if parser.peek_is("with") {
|
|
32
|
-
parser.advance(); // consume 'with'
|
|
33
|
-
if let Some(tok2) = parser.peek_clone() {
|
|
34
|
-
// target can be identifier or dotted identifier
|
|
35
|
-
if tok2.kind == TokenKind::Identifier {
|
|
36
|
-
parser.advance();
|
|
37
|
-
let mut base = tok2.lexeme.clone();
|
|
38
|
-
if let Some(dot) = parser.peek_clone() {
|
|
39
|
-
if dot.kind == TokenKind::Dot {
|
|
40
|
-
parser.advance();
|
|
41
|
-
if let Some(suf) = parser.peek_clone() {
|
|
42
|
-
if suf.kind == TokenKind::Identifier || suf.kind == TokenKind::Number {
|
|
43
|
-
parser.advance();
|
|
44
|
-
base.push('.');
|
|
45
|
-
base.push_str(&suf.lexeme);
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
}
|
|
49
|
-
}
|
|
50
|
-
target = Some(base);
|
|
51
|
-
}
|
|
52
|
-
}
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
// optional inline options map like { swing: 0.08 }
|
|
56
|
-
let mut options: Option<Value> = None;
|
|
57
|
-
if parser.check_token(TokenKind::LBrace) {
|
|
58
|
-
options = parser.parse_map_value();
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
// optional '=' and pattern string
|
|
62
|
-
let mut value: Value = Value::Null;
|
|
63
|
-
if parser.peek_is("=") {
|
|
64
|
-
parser.advance();
|
|
65
|
-
if let Some(tok3) = parser.peek_clone() {
|
|
66
|
-
if tok3.kind == TokenKind::String {
|
|
67
|
-
parser.advance();
|
|
68
|
-
let pat_str = Value::String(tok3.lexeme.clone());
|
|
69
|
-
if let Some(opts) = options {
|
|
70
|
-
// merge options map with pattern key
|
|
71
|
-
if let Value::Map(mut m) = opts {
|
|
72
|
-
m.insert("pattern".to_string(), pat_str);
|
|
73
|
-
value = Value::Map(m);
|
|
74
|
-
} else {
|
|
75
|
-
// unexpected, wrap into a map
|
|
76
|
-
let mut m = std::collections::HashMap::new();
|
|
77
|
-
m.insert("pattern".to_string(), pat_str);
|
|
78
|
-
value = Value::Map(m);
|
|
79
|
-
}
|
|
80
|
-
} else {
|
|
81
|
-
value = Value::String(tok3.lexeme.clone());
|
|
82
|
-
}
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
} else if let Some(opts) = options {
|
|
86
|
-
// only options were provided, store them as the value
|
|
87
|
-
value = opts;
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
Statement {
|
|
91
|
-
kind: StatementKind::Pattern { name, target },
|
|
92
|
-
value,
|
|
93
|
-
indent: tok.indent,
|
|
94
|
-
line: tok.line,
|
|
95
|
-
column: tok.column,
|
|
96
|
-
}
|
|
97
|
-
}
|