@devaloop/devalang 0.0.1-beta.1 → 0.0.1-beta.3
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 +84 -80
- package/README.md +10 -7
- package/docs/CHANGELOG.md +83 -0
- package/docs/ROADMAP.md +6 -2
- package/docs/TODO.md +3 -14
- package/examples/bus.deva +10 -0
- package/examples/chain.deva +19 -0
- package/examples/effect.deva +2 -0
- package/examples/filter.deva +11 -0
- package/examples/lfo.deva +9 -0
- package/examples/plugin.deva +10 -10
- package/examples/routing.deva +23 -0
- package/examples/synth.deva +11 -1
- package/examples/synth_types.deva +17 -0
- package/out-tsc/bin/project-version.json +6 -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 +1 -0
- package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
- package/out-tsc/scripts/version/copy-to-binary.js +79 -0
- package/package.json +23 -10
- package/project-version.json +3 -3
- package/rust/bindings/Cargo.toml +9 -0
- package/rust/bindings/src/lib.rs +86 -0
- package/rust/cli/addon/commands.rs +35 -0
- package/rust/cli/addon/download.rs +234 -0
- package/rust/cli/addon/install.rs +33 -0
- package/rust/cli/addon/list.rs +224 -0
- package/rust/cli/addon/metadata.rs +124 -0
- package/rust/cli/addon/mod.rs +8 -0
- package/rust/cli/addon/remove.rs +271 -0
- package/rust/cli/addon/update.rs +305 -0
- package/rust/cli/{install/addon.rs → addon/utils.rs} +34 -43
- package/rust/cli/build/commands.rs +153 -103
- package/rust/cli/build/mod.rs +2 -2
- package/rust/cli/build/process.rs +165 -146
- package/rust/cli/check/mod.rs +208 -208
- package/rust/cli/discover/commands.rs +53 -31
- package/rust/cli/discover/config.rs +2 -4
- package/rust/cli/discover/install.rs +139 -28
- package/rust/cli/discover/metadata.rs +3 -3
- package/rust/cli/login/commands.rs +124 -124
- package/rust/cli/me/commands.rs +52 -0
- package/rust/cli/me/mod.rs +1 -0
- package/rust/cli/mod.rs +2 -2
- package/rust/cli/parser.rs +76 -70
- package/rust/cli/play/commands.rs +375 -324
- package/rust/cli/play/mod.rs +5 -5
- package/rust/cli/play/process.rs +159 -150
- 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 +237 -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 +56 -7
- package/rust/core/audio/engine/notes/dsp.rs +88 -0
- package/rust/core/audio/engine/notes/mod.rs +53 -0
- package/rust/core/audio/engine/notes/params.rs +294 -0
- package/rust/core/audio/engine/sample/insert.rs +300 -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} +152 -310
- 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 +574 -542
- package/rust/core/audio/interpreter/mod.rs +2 -14
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +179 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +3 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +371 -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} +2 -4
- package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +36 -5
- package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -71
- package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +24 -26
- package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +36 -38
- package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +17 -19
- package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +115 -114
- 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} +54 -4
- package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
- package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +242 -239
- package/rust/core/audio/loader/trigger.rs +98 -97
- package/rust/core/audio/mod.rs +6 -7
- 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/modulator.rs +143 -143
- package/rust/core/builder/mod.rs +129 -86
- package/rust/core/debugger/{module.rs → logs.rs} +52 -55
- package/rust/core/debugger/mod.rs +30 -30
- package/rust/core/debugger/store.rs +38 -40
- package/rust/core/error/mod.rs +269 -269
- package/rust/core/lexer/driver.rs +2 -4
- package/rust/core/mod.rs +9 -10
- 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 +159 -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 +247 -0
- package/rust/core/parser/driver/parser.rs +160 -0
- package/rust/core/parser/handler/arrow_call.rs +90 -15
- 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 +39 -3
- package/rust/core/parser/handler/loop_.rs +194 -194
- package/rust/core/parser/handler/pattern.rs +25 -2
- package/rust/core/parser/handler/tempo.rs +105 -57
- package/rust/core/parser/statement.rs +10 -11
- package/rust/core/plugin/loader.rs +137 -137
- package/rust/core/plugin/runner/mod.rs +11 -0
- package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +206 -72
- package/rust/core/plugin/runner/wasm32.rs +44 -0
- package/rust/core/preprocessor/loader/inject.rs +313 -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 +55 -60
- package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +107 -114
- package/rust/core/preprocessor/processor/mod.rs +1 -0
- package/rust/core/preprocessor/resolver/function.rs +69 -69
- package/rust/core/preprocessor/resolver/group.rs +122 -94
- package/rust/core/preprocessor/resolver/pattern.rs +14 -2
- package/rust/core/store/global.rs +57 -61
- package/rust/core/store/mod.rs +1 -5
- package/rust/lib.rs +323 -308
- package/rust/macros/Cargo.toml +14 -0
- package/rust/macros/src/lib.rs +52 -0
- package/rust/main.rs +336 -143
- package/rust/types/Cargo.toml +1 -1
- package/rust/types/src/addons.rs +57 -55
- package/rust/types/src/config.rs +82 -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 +5 -2
- package/rust/utils/src/file.rs +477 -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 +158 -88
- package/rust/utils/src/signature.rs +41 -41
- package/rust/utils/src/spinner.rs +20 -20
- package/rust/utils/src/version.rs +58 -27
- package/rust/utils/src/watcher.rs +46 -46
- package/rust/web/api.rs +5 -5
- package/rust/web/auth.rs +5 -0
- package/rust/web/cdn.rs +34 -34
- package/rust/web/forge.rs +5 -0
- package/rust/web/mod.rs +2 -0
- package/tests/integration.rs +21 -21
- package/typescript/core/functions/index.ts +11 -0
- package/typescript/pkg/devalang_core.ts +20 -4
- package/typescript/scripts/version/copy-to-binary.ts +82 -0
- package/rust/cli/bank/api.rs +0 -122
- package/rust/cli/bank/commands.rs +0 -275
- package/rust/cli/bank/mod.rs +0 -29
- package/rust/cli/install/bank.rs +0 -53
- package/rust/cli/install/commands.rs +0 -35
- package/rust/cli/install/mod.rs +0 -4
- package/rust/cli/install/plugin.rs +0 -61
- 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,91 +1,91 @@
|
|
|
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
|
-
|
|
11
|
-
pub fn parse_call_token(
|
|
12
|
-
parser: &mut Parser,
|
|
13
|
-
current_token: Token,
|
|
14
|
-
_global_store: &mut GlobalStore,
|
|
15
|
-
) -> Statement {
|
|
16
|
-
parser.advance(); // consume "call"
|
|
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 'call'".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 call arguments".to_string(),
|
|
74
|
-
);
|
|
75
|
-
}
|
|
76
|
-
}
|
|
77
|
-
}
|
|
78
|
-
}
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
Statement {
|
|
82
|
-
kind: StatementKind::Call {
|
|
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
|
+
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_call_token(
|
|
12
|
+
parser: &mut Parser,
|
|
13
|
+
current_token: Token,
|
|
14
|
+
_global_store: &mut GlobalStore,
|
|
15
|
+
) -> Statement {
|
|
16
|
+
parser.advance(); // consume "call"
|
|
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 'call'".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 call arguments".to_string(),
|
|
74
|
+
);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
Statement {
|
|
82
|
+
kind: StatementKind::Call {
|
|
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,70 +1,70 @@
|
|
|
1
|
-
use crate::core::{
|
|
2
|
-
lexer::token::TokenKind,
|
|
3
|
-
parser::{
|
|
4
|
-
driver::Parser,
|
|
5
|
-
statement::{Statement, StatementKind},
|
|
6
|
-
},
|
|
7
|
-
store::global::GlobalStore,
|
|
8
|
-
};
|
|
9
|
-
use devalang_types::Value;
|
|
10
|
-
|
|
11
|
-
pub fn parse_emit_token(
|
|
12
|
-
parser: &mut Parser,
|
|
13
|
-
current: crate::core::lexer::token::Token,
|
|
14
|
-
_global_store: &mut GlobalStore,
|
|
15
|
-
) -> Statement {
|
|
16
|
-
parser.advance(); // consume 'emit'
|
|
17
|
-
|
|
18
|
-
let Some(ev) = parser.peek_clone() else {
|
|
19
|
-
return crate::core::parser::statement::error_from_token(
|
|
20
|
-
current,
|
|
21
|
-
"Expected event name after 'emit'".into(),
|
|
22
|
-
);
|
|
23
|
-
};
|
|
24
|
-
if ev.kind != TokenKind::Identifier {
|
|
25
|
-
return crate::core::parser::statement::error_from_token(
|
|
26
|
-
ev.clone(),
|
|
27
|
-
"Expected identifier as event name".into(),
|
|
28
|
-
);
|
|
29
|
-
}
|
|
30
|
-
let event_name = ev.lexeme.clone();
|
|
31
|
-
parser.advance(); // consume event name
|
|
32
|
-
|
|
33
|
-
// Optional payload on same line: number|string|identifier|map|array
|
|
34
|
-
let mut payload: Option<Value> = None;
|
|
35
|
-
if let Some(tok) = parser.peek_clone() {
|
|
36
|
-
if tok.line == ev.line {
|
|
37
|
-
let val = match tok.kind {
|
|
38
|
-
TokenKind::String => {
|
|
39
|
-
parser.advance();
|
|
40
|
-
Value::String(tok.lexeme.clone())
|
|
41
|
-
}
|
|
42
|
-
TokenKind::Number => {
|
|
43
|
-
parser.advance();
|
|
44
|
-
Value::Number(tok.lexeme.parse().unwrap_or(0.0))
|
|
45
|
-
}
|
|
46
|
-
TokenKind::Identifier => {
|
|
47
|
-
parser.advance();
|
|
48
|
-
Value::Identifier(tok.lexeme.clone())
|
|
49
|
-
}
|
|
50
|
-
TokenKind::LBrace => parser.parse_map_value().unwrap_or(Value::Null),
|
|
51
|
-
TokenKind::LBracket => parser.parse_array_value().unwrap_or(Value::Null),
|
|
52
|
-
_ => Value::Null,
|
|
53
|
-
};
|
|
54
|
-
if val != Value::Null {
|
|
55
|
-
payload = Some(val);
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
Statement {
|
|
61
|
-
kind: StatementKind::Emit {
|
|
62
|
-
event: event_name,
|
|
63
|
-
payload: payload.clone(),
|
|
64
|
-
},
|
|
65
|
-
value: payload.unwrap_or(Value::Null),
|
|
66
|
-
indent: current.indent,
|
|
67
|
-
line: current.line,
|
|
68
|
-
column: current.column,
|
|
69
|
-
}
|
|
70
|
-
}
|
|
1
|
+
use crate::core::{
|
|
2
|
+
lexer::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_emit_token(
|
|
12
|
+
parser: &mut Parser,
|
|
13
|
+
current: crate::core::lexer::token::Token,
|
|
14
|
+
_global_store: &mut GlobalStore,
|
|
15
|
+
) -> Statement {
|
|
16
|
+
parser.advance(); // consume 'emit'
|
|
17
|
+
|
|
18
|
+
let Some(ev) = parser.peek_clone() else {
|
|
19
|
+
return crate::core::parser::statement::error_from_token(
|
|
20
|
+
current,
|
|
21
|
+
"Expected event name after 'emit'".into(),
|
|
22
|
+
);
|
|
23
|
+
};
|
|
24
|
+
if ev.kind != TokenKind::Identifier {
|
|
25
|
+
return crate::core::parser::statement::error_from_token(
|
|
26
|
+
ev.clone(),
|
|
27
|
+
"Expected identifier as event name".into(),
|
|
28
|
+
);
|
|
29
|
+
}
|
|
30
|
+
let event_name = ev.lexeme.clone();
|
|
31
|
+
parser.advance(); // consume event name
|
|
32
|
+
|
|
33
|
+
// Optional payload on same line: number|string|identifier|map|array
|
|
34
|
+
let mut payload: Option<Value> = None;
|
|
35
|
+
if let Some(tok) = parser.peek_clone() {
|
|
36
|
+
if tok.line == ev.line {
|
|
37
|
+
let val = match tok.kind {
|
|
38
|
+
TokenKind::String => {
|
|
39
|
+
parser.advance();
|
|
40
|
+
Value::String(tok.lexeme.clone())
|
|
41
|
+
}
|
|
42
|
+
TokenKind::Number => {
|
|
43
|
+
parser.advance();
|
|
44
|
+
Value::Number(tok.lexeme.parse().unwrap_or(0.0))
|
|
45
|
+
}
|
|
46
|
+
TokenKind::Identifier => {
|
|
47
|
+
parser.advance();
|
|
48
|
+
Value::Identifier(tok.lexeme.clone())
|
|
49
|
+
}
|
|
50
|
+
TokenKind::LBrace => parser.parse_map_value().unwrap_or(Value::Null),
|
|
51
|
+
TokenKind::LBracket => parser.parse_array_value().unwrap_or(Value::Null),
|
|
52
|
+
_ => Value::Null,
|
|
53
|
+
};
|
|
54
|
+
if val != Value::Null {
|
|
55
|
+
payload = Some(val);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
Statement {
|
|
61
|
+
kind: StatementKind::Emit {
|
|
62
|
+
event: event_name,
|
|
63
|
+
payload: payload.clone(),
|
|
64
|
+
},
|
|
65
|
+
value: payload.unwrap_or(Value::Null),
|
|
66
|
+
indent: current.indent,
|
|
67
|
+
line: current.line,
|
|
68
|
+
column: current.column,
|
|
69
|
+
}
|
|
70
|
+
}
|
|
@@ -1,113 +1,113 @@
|
|
|
1
|
-
use devalang_types::Value;
|
|
2
|
-
|
|
3
|
-
use crate::core::{
|
|
4
|
-
lexer::token::TokenKind,
|
|
5
|
-
parser::{
|
|
6
|
-
driver::Parser,
|
|
7
|
-
statement::{Statement, StatementKind},
|
|
8
|
-
},
|
|
9
|
-
store::global::GlobalStore,
|
|
10
|
-
};
|
|
11
|
-
|
|
12
|
-
pub fn parse_function_token(parser: &mut Parser, global_store: &mut GlobalStore) -> Statement {
|
|
13
|
-
parser.advance(); // consume 'fn'
|
|
14
|
-
|
|
15
|
-
let fn_token = match parser.previous_clone() {
|
|
16
|
-
Some(tok) => tok,
|
|
17
|
-
None => return Statement::unknown(),
|
|
18
|
-
};
|
|
19
|
-
|
|
20
|
-
let name_token = match parser.peek_clone() {
|
|
21
|
-
Some(tok) => tok,
|
|
22
|
-
None => {
|
|
23
|
-
return crate::core::parser::statement::error_from_token(
|
|
24
|
-
fn_token,
|
|
25
|
-
"Expected function name after 'fn'".to_string(),
|
|
26
|
-
);
|
|
27
|
-
}
|
|
28
|
-
};
|
|
29
|
-
|
|
30
|
-
if name_token.kind != TokenKind::Identifier {
|
|
31
|
-
return crate::core::parser::statement::error_from_token(
|
|
32
|
-
name_token.clone(),
|
|
33
|
-
"Expected function name to be an identifier".to_string(),
|
|
34
|
-
);
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
let function_name = name_token.lexeme.clone();
|
|
38
|
-
parser.advance(); // consume function name
|
|
39
|
-
|
|
40
|
-
let mut parameters = Vec::new();
|
|
41
|
-
|
|
42
|
-
// Expect '('
|
|
43
|
-
if parser.peek_kind() != Some(TokenKind::LParen) {
|
|
44
|
-
return crate::core::parser::statement::error_from_token(
|
|
45
|
-
name_token.clone(),
|
|
46
|
-
"Expected '(' after function name".to_string(),
|
|
47
|
-
);
|
|
48
|
-
}
|
|
49
|
-
parser.advance(); // consume '('
|
|
50
|
-
|
|
51
|
-
// Parse parameters until ')'
|
|
52
|
-
let tokens = parser.collect_until(|t| t.kind == TokenKind::RParen || t.kind == TokenKind::EOF);
|
|
53
|
-
for token in tokens {
|
|
54
|
-
if token.kind == TokenKind::Identifier {
|
|
55
|
-
parameters.push(token.lexeme.clone());
|
|
56
|
-
}
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
if parser.peek_kind() == Some(TokenKind::RParen) {
|
|
60
|
-
parser.advance(); // consume ')'
|
|
61
|
-
} else {
|
|
62
|
-
return crate::core::parser::statement::error_from_token(
|
|
63
|
-
name_token.clone(),
|
|
64
|
-
"Expected ')' after parameters".to_string(),
|
|
65
|
-
);
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
// Expect colon
|
|
69
|
-
if parser.peek_kind() != Some(TokenKind::Colon) {
|
|
70
|
-
return crate::core::parser::statement::error_from_token(
|
|
71
|
-
name_token.clone(),
|
|
72
|
-
"Expected ':' after ')'".to_string(),
|
|
73
|
-
);
|
|
74
|
-
}
|
|
75
|
-
parser.advance(); // consume ':'
|
|
76
|
-
|
|
77
|
-
// Collect ALL tokens indented after this line until Dedent
|
|
78
|
-
let base_indent = fn_token.indent;
|
|
79
|
-
let mut body_tokens = Vec::new();
|
|
80
|
-
|
|
81
|
-
while let Some(tok) = parser.peek() {
|
|
82
|
-
if tok.kind == TokenKind::Dedent && tok.indent <= base_indent {
|
|
83
|
-
break;
|
|
84
|
-
}
|
|
85
|
-
if let Some(t) = parser.advance() {
|
|
86
|
-
body_tokens.push(t.clone());
|
|
87
|
-
} else {
|
|
88
|
-
break;
|
|
89
|
-
}
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
// Parse those tokens into block statements
|
|
93
|
-
let body = parser.parse_block(body_tokens.clone(), global_store);
|
|
94
|
-
|
|
95
|
-
// Skip Dedent if present
|
|
96
|
-
if let Some(tok) = parser.peek() {
|
|
97
|
-
if tok.kind == TokenKind::Dedent {
|
|
98
|
-
parser.advance();
|
|
99
|
-
}
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
Statement {
|
|
103
|
-
kind: StatementKind::Function {
|
|
104
|
-
name: function_name.clone(),
|
|
105
|
-
parameters: parameters.clone(),
|
|
106
|
-
body: body.clone(),
|
|
107
|
-
},
|
|
108
|
-
value: Value::Null,
|
|
109
|
-
indent: fn_token.indent,
|
|
110
|
-
line: fn_token.line,
|
|
111
|
-
column: fn_token.column,
|
|
112
|
-
}
|
|
113
|
-
}
|
|
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_function_token(parser: &mut Parser, global_store: &mut GlobalStore) -> Statement {
|
|
13
|
+
parser.advance(); // consume 'fn'
|
|
14
|
+
|
|
15
|
+
let fn_token = match parser.previous_clone() {
|
|
16
|
+
Some(tok) => tok,
|
|
17
|
+
None => return Statement::unknown(),
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
let name_token = match parser.peek_clone() {
|
|
21
|
+
Some(tok) => tok,
|
|
22
|
+
None => {
|
|
23
|
+
return crate::core::parser::statement::error_from_token(
|
|
24
|
+
fn_token,
|
|
25
|
+
"Expected function name after 'fn'".to_string(),
|
|
26
|
+
);
|
|
27
|
+
}
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
if name_token.kind != TokenKind::Identifier {
|
|
31
|
+
return crate::core::parser::statement::error_from_token(
|
|
32
|
+
name_token.clone(),
|
|
33
|
+
"Expected function name to be an identifier".to_string(),
|
|
34
|
+
);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
let function_name = name_token.lexeme.clone();
|
|
38
|
+
parser.advance(); // consume function name
|
|
39
|
+
|
|
40
|
+
let mut parameters = Vec::new();
|
|
41
|
+
|
|
42
|
+
// Expect '('
|
|
43
|
+
if parser.peek_kind() != Some(TokenKind::LParen) {
|
|
44
|
+
return crate::core::parser::statement::error_from_token(
|
|
45
|
+
name_token.clone(),
|
|
46
|
+
"Expected '(' after function name".to_string(),
|
|
47
|
+
);
|
|
48
|
+
}
|
|
49
|
+
parser.advance(); // consume '('
|
|
50
|
+
|
|
51
|
+
// Parse parameters until ')'
|
|
52
|
+
let tokens = parser.collect_until(|t| t.kind == TokenKind::RParen || t.kind == TokenKind::EOF);
|
|
53
|
+
for token in tokens {
|
|
54
|
+
if token.kind == TokenKind::Identifier {
|
|
55
|
+
parameters.push(token.lexeme.clone());
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if parser.peek_kind() == Some(TokenKind::RParen) {
|
|
60
|
+
parser.advance(); // consume ')'
|
|
61
|
+
} else {
|
|
62
|
+
return crate::core::parser::statement::error_from_token(
|
|
63
|
+
name_token.clone(),
|
|
64
|
+
"Expected ')' after parameters".to_string(),
|
|
65
|
+
);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// Expect colon
|
|
69
|
+
if parser.peek_kind() != Some(TokenKind::Colon) {
|
|
70
|
+
return crate::core::parser::statement::error_from_token(
|
|
71
|
+
name_token.clone(),
|
|
72
|
+
"Expected ':' after ')'".to_string(),
|
|
73
|
+
);
|
|
74
|
+
}
|
|
75
|
+
parser.advance(); // consume ':'
|
|
76
|
+
|
|
77
|
+
// Collect ALL tokens indented after this line until Dedent
|
|
78
|
+
let base_indent = fn_token.indent;
|
|
79
|
+
let mut body_tokens = Vec::new();
|
|
80
|
+
|
|
81
|
+
while let Some(tok) = parser.peek() {
|
|
82
|
+
if tok.kind == TokenKind::Dedent && tok.indent <= base_indent {
|
|
83
|
+
break;
|
|
84
|
+
}
|
|
85
|
+
if let Some(t) = parser.advance() {
|
|
86
|
+
body_tokens.push(t.clone());
|
|
87
|
+
} else {
|
|
88
|
+
break;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Parse those tokens into block statements
|
|
93
|
+
let body = parser.parse_block(body_tokens.clone(), global_store);
|
|
94
|
+
|
|
95
|
+
// Skip Dedent if present
|
|
96
|
+
if let Some(tok) = parser.peek() {
|
|
97
|
+
if tok.kind == TokenKind::Dedent {
|
|
98
|
+
parser.advance();
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
Statement {
|
|
103
|
+
kind: StatementKind::Function {
|
|
104
|
+
name: function_name.clone(),
|
|
105
|
+
parameters: parameters.clone(),
|
|
106
|
+
body: body.clone(),
|
|
107
|
+
},
|
|
108
|
+
value: Value::Null,
|
|
109
|
+
indent: fn_token.indent,
|
|
110
|
+
line: fn_token.line,
|
|
111
|
+
column: fn_token.column,
|
|
112
|
+
}
|
|
113
|
+
}
|