@devaloop/devalang 0.0.1-alpha.8 → 0.0.1-beta.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/.cargo/config.toml +2 -0
- package/.devalang +10 -4
- package/.github/workflows/ci.yml +103 -0
- package/Cargo.toml +80 -48
- package/README.md +135 -158
- package/docs/CHANGELOG.md +413 -1
- package/docs/CONTRIBUTING.md +101 -0
- package/docs/ROADMAP.md +10 -7
- package/docs/TODO.md +21 -9
- package/examples/automation.deva +42 -0
- package/examples/bank.deva +7 -0
- package/examples/condition.deva +8 -12
- package/examples/duration.deva +9 -0
- package/examples/events.deva +12 -0
- package/examples/function.deva +15 -0
- package/examples/group.deva +3 -3
- package/examples/index.deva +57 -10
- package/examples/loop.deva +7 -12
- package/examples/pattern.deva +8 -0
- package/examples/plugin.deva +16 -0
- package/examples/synth.deva +14 -0
- package/examples/variables.deva +2 -2
- package/out-tsc/bin/index.d.ts +2 -0
- package/out-tsc/bin/index.js +51 -7
- package/out-tsc/core/functions/index.d.ts +37 -0
- package/out-tsc/core/functions/index.js +76 -0
- package/out-tsc/core/index.d.ts +6 -0
- package/out-tsc/core/index.js +22 -0
- package/out-tsc/core/types/index.d.ts +4 -0
- package/out-tsc/core/types/index.js +20 -0
- package/out-tsc/core/types/plugin.d.ts +18 -0
- package/out-tsc/core/types/plugin.js +2 -0
- package/out-tsc/core/types/result.d.ts +27 -0
- package/out-tsc/core/types/result.js +2 -0
- package/out-tsc/core/types/statement.d.ts +106 -0
- package/out-tsc/core/types/statement.js +2 -0
- package/out-tsc/core/types/value.d.ts +43 -0
- package/out-tsc/core/types/value.js +2 -0
- package/out-tsc/index.d.ts +7 -0
- package/out-tsc/index.js +42 -1
- package/out-tsc/pkg/devalang_core.d.ts +13 -0
- package/out-tsc/pkg/devalang_core.js +50 -0
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +33 -0
- package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
- package/out-tsc/scripts/copy-wasm-dts.js +73 -0
- package/out-tsc/scripts/postinstall.d.ts +1 -0
- package/out-tsc/scripts/postinstall.js +83 -0
- package/out-tsc/scripts/version/bump.d.ts +1 -0
- package/out-tsc/scripts/version/fetch.d.ts +1 -0
- package/out-tsc/scripts/version/fetch.js +1 -5
- package/out-tsc/scripts/version/index.d.ts +1 -0
- package/out-tsc/scripts/version/sync.d.ts +1 -0
- package/package.json +28 -7
- package/project-version.json +4 -4
- package/rust/cli/bank/api.rs +122 -0
- package/rust/cli/bank/commands.rs +275 -0
- package/rust/cli/bank/mod.rs +29 -0
- package/rust/cli/build/commands.rs +103 -0
- package/rust/cli/build/mod.rs +2 -0
- package/rust/cli/build/process.rs +146 -0
- package/rust/cli/check/mod.rs +208 -0
- package/rust/cli/discover/commands.rs +253 -0
- package/rust/cli/discover/config.rs +111 -0
- package/rust/cli/discover/fs.rs +19 -0
- package/rust/cli/discover/install.rs +103 -0
- package/rust/cli/discover/metadata.rs +48 -0
- package/rust/cli/discover/mod.rs +5 -0
- package/rust/cli/{init.rs → init/commands.rs} +32 -23
- package/rust/cli/init/mod.rs +1 -0
- package/rust/cli/install/addon.rs +118 -0
- package/rust/cli/install/bank.rs +53 -0
- package/rust/cli/install/commands.rs +35 -0
- package/rust/cli/install/mod.rs +4 -0
- package/rust/cli/install/plugin.rs +61 -0
- package/rust/cli/login/commands.rs +124 -0
- package/rust/cli/login/mod.rs +1 -0
- package/rust/cli/mod.rs +12 -205
- package/rust/cli/parser.rs +314 -0
- package/rust/cli/play/commands.rs +324 -0
- package/rust/cli/play/io.rs +17 -0
- package/rust/cli/play/mod.rs +5 -0
- package/rust/cli/play/process.rs +150 -0
- package/rust/cli/play/realtime.rs +91 -0
- package/rust/cli/play/utils.rs +23 -0
- package/rust/cli/telemetry/commands.rs +22 -0
- package/rust/cli/telemetry/event_creator.rs +80 -0
- package/rust/cli/telemetry/mod.rs +3 -0
- package/rust/cli/telemetry/send.rs +51 -0
- package/rust/cli/{template.rs → template/commands.rs} +69 -57
- package/rust/cli/template/mod.rs +1 -0
- package/rust/cli/update/commands.rs +6 -0
- package/rust/cli/update/mod.rs +1 -0
- package/rust/config/driver.rs +103 -0
- package/rust/config/mod.rs +3 -16
- package/rust/config/ops.rs +26 -0
- package/rust/config/settings.rs +101 -0
- package/rust/core/audio/engine/helpers.rs +170 -0
- package/rust/core/audio/engine/mod.rs +7 -0
- package/rust/core/audio/engine/sample.rs +366 -0
- package/rust/core/audio/engine/synth.rs +325 -0
- package/rust/core/audio/evaluator.rs +310 -31
- package/rust/core/audio/interpreter/arrow_call.rs +311 -0
- package/rust/core/audio/interpreter/automate.rs +18 -0
- package/rust/core/audio/interpreter/call.rs +294 -42
- package/rust/core/audio/interpreter/condition.rs +71 -65
- package/rust/core/audio/interpreter/driver.rs +542 -204
- package/rust/core/audio/interpreter/function.rs +26 -0
- package/rust/core/audio/interpreter/let_.rs +38 -19
- package/rust/core/audio/interpreter/load.rs +19 -18
- package/rust/core/audio/interpreter/loop_.rs +114 -59
- package/rust/core/audio/interpreter/mod.rs +14 -11
- package/rust/core/audio/interpreter/sleep.rs +28 -36
- package/rust/core/audio/interpreter/spawn.rs +252 -65
- package/rust/core/audio/interpreter/tempo.rs +40 -16
- package/rust/core/audio/interpreter/trigger.rs +239 -69
- package/rust/core/audio/loader/mod.rs +1 -1
- package/rust/core/audio/loader/trigger.rs +97 -52
- package/rust/core/audio/mod.rs +7 -6
- package/rust/core/audio/player.rs +70 -54
- package/rust/core/audio/renderer.rs +54 -57
- package/rust/core/audio/special/easing.rs +189 -0
- package/rust/core/audio/special/env.rs +45 -0
- package/rust/core/audio/special/math.rs +134 -0
- package/rust/core/audio/special/mod.rs +9 -0
- package/rust/core/audio/special/modulator.rs +143 -0
- package/rust/core/builder/mod.rs +86 -80
- package/rust/core/debugger/lexer.rs +27 -27
- package/rust/core/debugger/mod.rs +30 -21
- package/rust/core/debugger/module.rs +55 -0
- package/rust/core/debugger/preprocessor.rs +27 -27
- package/rust/core/debugger/store.rs +40 -25
- package/rust/core/error/mod.rs +269 -60
- package/rust/core/lexer/driver.rs +61 -0
- package/rust/core/lexer/handler/arrow.rs +82 -0
- 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 -215
- package/rust/core/lexer/handler/identifier.rs +47 -40
- package/rust/core/lexer/handler/indent.rs +66 -52
- package/rust/core/lexer/handler/mod.rs +15 -13
- 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 -44
- package/rust/core/lexer/handler/parenthesis.rs +41 -0
- package/rust/core/lexer/handler/slash.rs +21 -0
- package/rust/core/lexer/handler/string.rs +63 -63
- package/rust/core/lexer/mod.rs +3 -30
- package/rust/core/lexer/token.rs +21 -12
- package/rust/core/mod.rs +10 -10
- package/rust/core/parser/driver.rs +584 -312
- package/rust/core/parser/handler/arrow_call.rs +253 -0
- package/rust/core/parser/handler/at.rs +279 -162
- package/rust/core/parser/handler/bank.rs +104 -41
- package/rust/core/parser/handler/condition.rs +83 -74
- package/rust/core/parser/handler/dot.rs +148 -112
- package/rust/core/parser/handler/identifier/automate.rs +254 -0
- package/rust/core/parser/handler/identifier/call.rs +91 -0
- package/rust/core/parser/handler/identifier/emit.rs +70 -0
- package/rust/core/parser/handler/identifier/function.rs +113 -0
- package/rust/core/parser/handler/identifier/group.rs +89 -0
- package/rust/core/parser/handler/identifier/let_.rs +173 -0
- package/rust/core/parser/handler/identifier/mod.rs +55 -0
- package/rust/core/parser/handler/identifier/on.rs +107 -0
- package/rust/core/parser/handler/identifier/print.rs +49 -0
- package/rust/core/parser/handler/identifier/sleep.rs +43 -0
- package/rust/core/parser/handler/identifier/spawn.rs +91 -0
- package/rust/core/parser/handler/identifier/synth.rs +135 -0
- package/rust/core/parser/handler/loop_.rs +194 -66
- package/rust/core/parser/handler/mod.rs +9 -7
- package/rust/core/parser/handler/pattern.rs +74 -0
- package/rust/core/parser/handler/tempo.rs +57 -47
- package/rust/core/parser/mod.rs +3 -4
- package/rust/core/parser/statement.rs +11 -88
- package/rust/core/plugin/loader.rs +137 -0
- package/rust/core/plugin/mod.rs +2 -0
- package/rust/core/plugin/runner.rs +347 -0
- package/rust/core/preprocessor/loader.rs +637 -179
- package/rust/core/preprocessor/mod.rs +4 -4
- package/rust/core/preprocessor/module.rs +60 -53
- package/rust/core/preprocessor/processor.rs +114 -67
- package/rust/core/preprocessor/resolver/bank.rs +49 -47
- package/rust/core/preprocessor/resolver/call.rs +124 -53
- package/rust/core/preprocessor/resolver/condition.rs +95 -66
- package/rust/core/preprocessor/resolver/driver.rs +324 -182
- package/rust/core/preprocessor/resolver/function.rs +69 -0
- package/rust/core/preprocessor/resolver/group.rs +94 -118
- package/rust/core/preprocessor/resolver/let_.rs +32 -0
- package/rust/core/preprocessor/resolver/loop_.rs +318 -145
- package/rust/core/preprocessor/resolver/mod.rs +16 -10
- package/rust/core/preprocessor/resolver/pattern.rs +83 -0
- package/rust/core/preprocessor/resolver/spawn.rs +99 -53
- package/rust/core/preprocessor/resolver/synth.rs +54 -0
- package/rust/core/preprocessor/resolver/tempo.rs +48 -49
- package/rust/core/preprocessor/resolver/trigger.rs +116 -111
- package/rust/core/preprocessor/resolver/value.rs +176 -0
- package/rust/core/store/export.rs +28 -28
- package/rust/core/store/function.rs +40 -0
- package/rust/core/store/global.rs +61 -39
- package/rust/core/store/import.rs +28 -28
- package/rust/core/store/mod.rs +5 -4
- package/rust/core/store/variable.rs +51 -28
- package/rust/core/utils/mod.rs +1 -2
- package/rust/core/utils/path.rs +37 -46
- package/rust/lib.rs +308 -117
- package/rust/main.rs +364 -65
- package/rust/types/Cargo.toml +11 -0
- package/rust/types/src/addons.rs +55 -0
- package/rust/types/src/ast.rs +202 -0
- package/rust/types/src/config.rs +74 -0
- package/rust/types/src/lib.rs +12 -0
- package/rust/types/src/telemetry.rs +85 -0
- package/rust/utils/Cargo.toml +26 -0
- package/rust/utils/src/error.rs +186 -0
- package/rust/utils/src/file.rs +94 -0
- package/rust/utils/src/first_usage.rs +97 -0
- package/rust/utils/{mod.rs → src/lib.rs} +9 -6
- package/rust/utils/{logger.rs → src/logger.rs} +200 -123
- package/rust/utils/src/path.rs +88 -0
- package/rust/utils/src/signature.rs +41 -0
- package/rust/utils/{spinner.rs → src/spinner.rs} +20 -21
- package/rust/utils/src/version.rs +27 -0
- package/rust/utils/{watcher.rs → src/watcher.rs} +46 -33
- package/rust/web/api.rs +5 -0
- package/rust/web/cdn.rs +34 -0
- package/rust/web/mod.rs +3 -0
- package/rust/web/sso.rs +5 -0
- package/templates/minimal/README.md +143 -127
- package/templates/welcome/README.md +143 -127
- package/templates/welcome/src/index.deva +56 -8
- package/templates/welcome/src/variables.deva +2 -4
- package/tests/integration.rs +21 -0
- package/tests/rust/cli_check_build.rs +21 -0
- package/tests/rust/cli_help.rs +12 -0
- package/tests/rust/cli_template_list.rs +10 -0
- package/tests/rust/cli_version.rs +11 -0
- package/tests/typescript/index.spec.ts +136 -0
- package/tests/typescript/playhead.spec.ts +36 -0
- package/tests/typescript/render_e2e.spec.ts +77 -0
- package/tsconfig.json +12 -10
- package/typescript/bin/index.ts +19 -5
- package/typescript/core/functions/index.ts +83 -0
- package/typescript/core/index.ts +6 -0
- package/typescript/core/types/index.ts +4 -0
- package/typescript/core/types/plugin.ts +19 -0
- package/typescript/core/types/result.ts +29 -0
- package/typescript/core/types/statement.ts +47 -0
- package/typescript/core/types/value.ts +29 -0
- package/typescript/index.ts +8 -1
- package/typescript/pkg/devalang_core.d.ts +4 -0
- package/typescript/pkg/devalang_core.ts +49 -0
- package/typescript/scripts/copy-wasm-dts.ts +41 -0
- package/typescript/scripts/postinstall.ts +85 -0
- package/typescript/scripts/version/bump.ts +0 -1
- package/typescript/scripts/version/fetch.ts +1 -6
- package/typescript/scripts/version/index.ts +0 -1
- package/docs/COMMANDS.md +0 -85
- package/docs/CONFIG.md +0 -30
- package/docs/SYNTAX.md +0 -210
- package/out-tsc/bin/devalang.exe +0 -0
- package/out-tsc/scripts/postbuild.js +0 -11
- package/rust/cli/build.rs +0 -137
- package/rust/cli/check.rs +0 -117
- package/rust/cli/play.rs +0 -193
- package/rust/config/loader.rs +0 -13
- package/rust/core/audio/engine.rs +0 -126
- package/rust/core/parser/handler/identifier.rs +0 -262
- package/rust/core/shared/duration.rs +0 -8
- package/rust/core/shared/mod.rs +0 -2
- package/rust/core/shared/value.rs +0 -18
- package/rust/core/utils/validation.rs +0 -35
- package/rust/utils/file.rs +0 -35
- package/rust/utils/signature.rs +0 -17
- package/rust/utils/version.rs +0 -15
- package/typescript/scripts/postbuild.ts +0 -8
|
@@ -0,0 +1,176 @@
|
|
|
1
|
+
use std::collections::HashMap;
|
|
2
|
+
use toml::Value as TomlValue;
|
|
3
|
+
|
|
4
|
+
use crate::core::{
|
|
5
|
+
preprocessor::{module::Module, resolver::driver::resolve_statement},
|
|
6
|
+
store::global::GlobalStore,
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
use devalang_types::Value;
|
|
10
|
+
|
|
11
|
+
fn find_export_value(name: &str, global_store: &GlobalStore) -> Option<Value> {
|
|
12
|
+
for module in global_store.modules.values() {
|
|
13
|
+
if let Some(val) = module.export_table.get_export(name) {
|
|
14
|
+
return Some(val.clone());
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
None
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
pub fn resolve_value(value: &Value, module: &Module, global_store: &mut GlobalStore) -> Value {
|
|
22
|
+
match value {
|
|
23
|
+
Value::String(s) => {
|
|
24
|
+
// Keep raw strings as-is; they may be runtime-evaluated (e.g., expressions)
|
|
25
|
+
Value::String(s.clone())
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
Value::Identifier(name) => {
|
|
29
|
+
if let Some(original_val) = module.variable_table.get(name) {
|
|
30
|
+
return resolve_value(original_val, module, global_store);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
if let Some(export_val) = find_export_value(name, global_store) {
|
|
34
|
+
return resolve_value(&export_val, module, global_store);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
// Leave unresolved identifiers as-is; may be runtime-bound (e.g., foreach variable)
|
|
38
|
+
Value::Identifier(name.clone())
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
Value::Map(map) => {
|
|
42
|
+
if let Some(Value::Identifier(entity)) = map.get("entity") {
|
|
43
|
+
// SECTION Synth
|
|
44
|
+
if entity == "synth" {
|
|
45
|
+
if let Some(Value::Map(synth_data)) = map.get("value") {
|
|
46
|
+
let resolved_waveform = synth_data
|
|
47
|
+
.get("waveform")
|
|
48
|
+
.map(|wf| resolve_value(wf, module, global_store))
|
|
49
|
+
.unwrap_or(Value::Null);
|
|
50
|
+
|
|
51
|
+
let resolved_params = synth_data
|
|
52
|
+
.get("parameters")
|
|
53
|
+
.map(|p| resolve_value(p, module, global_store))
|
|
54
|
+
.unwrap_or(Value::Map(HashMap::new()));
|
|
55
|
+
|
|
56
|
+
// If waveform refers to a plugin synth (e.g., alias.synth),
|
|
57
|
+
// merge plugin-exported defaults (dynamic) into parameters and
|
|
58
|
+
// allow 'waveform' override from parameters map.
|
|
59
|
+
let mut final_waveform = resolved_waveform.clone();
|
|
60
|
+
let mut params_map = match resolved_params.clone() {
|
|
61
|
+
Value::Map(m) => m,
|
|
62
|
+
_ => HashMap::new(),
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
// Helper: convert TomlValue into runtime Value
|
|
66
|
+
fn toml_to_value(tv: &TomlValue) -> Value {
|
|
67
|
+
match tv {
|
|
68
|
+
TomlValue::String(s) => Value::String(s.clone()),
|
|
69
|
+
TomlValue::Integer(i) => Value::Number(*i as f32),
|
|
70
|
+
TomlValue::Float(f) => Value::Number(*f as f32),
|
|
71
|
+
TomlValue::Boolean(b) => Value::Boolean(*b),
|
|
72
|
+
TomlValue::Array(arr) => {
|
|
73
|
+
Value::Array(arr.iter().map(toml_to_value).collect())
|
|
74
|
+
}
|
|
75
|
+
TomlValue::Table(t) => {
|
|
76
|
+
let mut m = HashMap::new();
|
|
77
|
+
for (k, v) in t.iter() {
|
|
78
|
+
m.insert(k.clone(), toml_to_value(v));
|
|
79
|
+
}
|
|
80
|
+
Value::Map(m)
|
|
81
|
+
}
|
|
82
|
+
_ => Value::Null,
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Detect plugin alias from waveform string like "alias.synth" OR just "alias"
|
|
87
|
+
let (alias_opt, explicit_synth_export) = match &final_waveform {
|
|
88
|
+
Value::String(s) | Value::Identifier(s) => {
|
|
89
|
+
let parts: Vec<&str> = s.split('.').collect();
|
|
90
|
+
if parts.len() >= 2 && parts[1] == "synth" {
|
|
91
|
+
(Some(parts[0].to_string()), true)
|
|
92
|
+
} else if parts.len() == 1 {
|
|
93
|
+
(Some(parts[0].to_string()), false)
|
|
94
|
+
} else {
|
|
95
|
+
(None, false)
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
_ => (None, false),
|
|
99
|
+
};
|
|
100
|
+
|
|
101
|
+
if let Some(alias) = alias_opt {
|
|
102
|
+
// Resolve alias -> plugin uri -> plugin info
|
|
103
|
+
if let Some(Value::String(uri)) = module.variable_table.get(&alias) {
|
|
104
|
+
if let Some(id) = uri.strip_prefix("devalang://plugin/") {
|
|
105
|
+
let mut parts = id.split('.');
|
|
106
|
+
let author = parts.next().unwrap_or("");
|
|
107
|
+
let pname = parts.next().unwrap_or("");
|
|
108
|
+
let key = format!("{}:{}", author, pname);
|
|
109
|
+
if let Some((plugin_info, _wasm)) =
|
|
110
|
+
global_store.plugins.get(&key)
|
|
111
|
+
{
|
|
112
|
+
// Merge defaults dynamically from exports
|
|
113
|
+
for exp in &plugin_info.exports {
|
|
114
|
+
// Skip entry named 'synth' which is used as the flag
|
|
115
|
+
if exp.name == "synth" {
|
|
116
|
+
continue;
|
|
117
|
+
}
|
|
118
|
+
if let Some(def) = &exp.default {
|
|
119
|
+
let val = toml_to_value(def);
|
|
120
|
+
// only apply if not overridden by user params
|
|
121
|
+
params_map.entry(exp.name.clone()).or_insert(val);
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
// If 'waveform' is provided in params (by user or default), use it
|
|
126
|
+
if let Some(wf_val) = params_map.remove("waveform") {
|
|
127
|
+
final_waveform =
|
|
128
|
+
resolve_value(&wf_val, module, global_store);
|
|
129
|
+
} else if let Some(wf_default) = plugin_info
|
|
130
|
+
.exports
|
|
131
|
+
.iter()
|
|
132
|
+
.find(|e| e.name == "waveform")
|
|
133
|
+
.and_then(|e| e.default.as_ref())
|
|
134
|
+
{
|
|
135
|
+
final_waveform = toml_to_value(wf_default);
|
|
136
|
+
} else if explicit_synth_export {
|
|
137
|
+
// keep as alias.synth if no default waveform
|
|
138
|
+
} else {
|
|
139
|
+
// If no explicit .synth in waveform, but alias is a plugin,
|
|
140
|
+
// treat it as alias.synth by default to enable plugin synth usage
|
|
141
|
+
final_waveform =
|
|
142
|
+
Value::String(format!("{}.synth", alias));
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
let mut result = HashMap::new();
|
|
150
|
+
result.insert("waveform".to_string(), final_waveform);
|
|
151
|
+
result.insert("parameters".to_string(), Value::Map(params_map));
|
|
152
|
+
|
|
153
|
+
return Value::Map(result);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
|
|
158
|
+
let mut resolved = HashMap::new();
|
|
159
|
+
for (k, v) in map {
|
|
160
|
+
resolved.insert(k.clone(), resolve_value(v, module, global_store));
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
Value::Map(resolved)
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
Value::Block(stmts) => {
|
|
167
|
+
let resolved_stmts = stmts
|
|
168
|
+
.iter()
|
|
169
|
+
.map(|stmt| resolve_statement(stmt, module, &module.path, global_store))
|
|
170
|
+
.collect();
|
|
171
|
+
Value::Block(resolved_stmts)
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
other => other.clone(),
|
|
175
|
+
}
|
|
176
|
+
}
|
|
@@ -1,28 +1,28 @@
|
|
|
1
|
-
use std::collections::HashMap;
|
|
2
|
-
|
|
3
|
-
use
|
|
4
|
-
|
|
5
|
-
#[derive(Debug, Default, Clone, PartialEq)]
|
|
6
|
-
pub struct ExportTable {
|
|
7
|
-
pub exports: HashMap<String, Value>,
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
impl ExportTable {
|
|
11
|
-
pub fn new() -> Self {
|
|
12
|
-
ExportTable {
|
|
13
|
-
exports: HashMap::new(),
|
|
14
|
-
}
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
pub fn add_export(&mut self, name: String, value: Value) {
|
|
18
|
-
self.exports.insert(name, value);
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
pub fn get_export(&self, name: &str) -> Option<&Value> {
|
|
22
|
-
self.exports.get(name)
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
pub fn remove_export(&mut self, name: &str) -> Option<Value> {
|
|
26
|
-
self.exports.remove(name)
|
|
27
|
-
}
|
|
28
|
-
}
|
|
1
|
+
use std::collections::HashMap;
|
|
2
|
+
|
|
3
|
+
use devalang_types::Value;
|
|
4
|
+
|
|
5
|
+
#[derive(Debug, Default, Clone, PartialEq)]
|
|
6
|
+
pub struct ExportTable {
|
|
7
|
+
pub exports: HashMap<String, Value>,
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
impl ExportTable {
|
|
11
|
+
pub fn new() -> Self {
|
|
12
|
+
ExportTable {
|
|
13
|
+
exports: HashMap::new(),
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
pub fn add_export(&mut self, name: String, value: Value) {
|
|
18
|
+
self.exports.insert(name, value);
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
pub fn get_export(&self, name: &str) -> Option<&Value> {
|
|
22
|
+
self.exports.get(name)
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
pub fn remove_export(&mut self, name: &str) -> Option<Value> {
|
|
26
|
+
self.exports.remove(name)
|
|
27
|
+
}
|
|
28
|
+
}
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
use crate::core::parser::statement::Statement;
|
|
2
|
+
use std::collections::HashMap;
|
|
3
|
+
|
|
4
|
+
#[derive(Debug, Clone)]
|
|
5
|
+
pub struct FunctionDef {
|
|
6
|
+
pub name: String,
|
|
7
|
+
pub parameters: Vec<String>,
|
|
8
|
+
pub body: Vec<Statement>,
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
#[derive(Debug, Clone)]
|
|
12
|
+
pub struct FunctionTable {
|
|
13
|
+
pub functions: HashMap<String, FunctionDef>,
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
impl Default for FunctionTable {
|
|
17
|
+
fn default() -> Self {
|
|
18
|
+
Self::new()
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
impl FunctionTable {
|
|
23
|
+
pub fn new() -> Self {
|
|
24
|
+
FunctionTable {
|
|
25
|
+
functions: HashMap::new(),
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
pub fn add_function(&mut self, function: FunctionDef) {
|
|
30
|
+
self.functions.insert(function.name.clone(), function);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
pub fn get_function(&self, name: &str) -> Option<&FunctionDef> {
|
|
34
|
+
self.functions.get(name)
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
pub fn remove_function(&mut self, name: &str) -> Option<FunctionDef> {
|
|
38
|
+
self.functions.remove(name)
|
|
39
|
+
}
|
|
40
|
+
}
|
|
@@ -1,39 +1,61 @@
|
|
|
1
|
-
use
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
1
|
+
use crate::core::{
|
|
2
|
+
parser::statement::Statement,
|
|
3
|
+
preprocessor::module::Module,
|
|
4
|
+
store::{function::FunctionTable, variable::VariableTable},
|
|
5
|
+
};
|
|
6
|
+
use devalang_types::PluginInfo;
|
|
7
|
+
use std::collections::HashMap;
|
|
8
|
+
|
|
9
|
+
#[derive(Debug, Clone)]
|
|
10
|
+
pub struct GlobalStore {
|
|
11
|
+
pub modules: HashMap<String, Module>,
|
|
12
|
+
pub variables: VariableTable,
|
|
13
|
+
pub functions: FunctionTable,
|
|
14
|
+
pub events: HashMap<String, Vec<Statement>>,
|
|
15
|
+
pub plugins: HashMap<String, (PluginInfo, Vec<u8>)>,
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
impl Default for GlobalStore {
|
|
19
|
+
fn default() -> Self {
|
|
20
|
+
Self::new()
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
impl GlobalStore {
|
|
25
|
+
pub fn new() -> Self {
|
|
26
|
+
GlobalStore {
|
|
27
|
+
modules: HashMap::new(),
|
|
28
|
+
functions: FunctionTable::new(),
|
|
29
|
+
variables: VariableTable::new(),
|
|
30
|
+
events: HashMap::new(),
|
|
31
|
+
plugins: HashMap::new(),
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
pub fn insert_module(&mut self, path: String, module: Module) {
|
|
36
|
+
self.modules.insert(path, module);
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
pub fn modules_mut(&mut self) -> &mut HashMap<String, Module> {
|
|
40
|
+
&mut self.modules
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
pub fn get_module(&self, path: &str) -> Option<&Module> {
|
|
44
|
+
self.modules.get(path)
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
pub fn remove_module(&mut self, path: &str) -> Option<Module> {
|
|
48
|
+
self.modules.remove(path)
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
pub fn register_event_handler(&mut self, event: &str, handler: Statement) {
|
|
52
|
+
self.events
|
|
53
|
+
.entry(event.to_string())
|
|
54
|
+
.or_default()
|
|
55
|
+
.push(handler);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
pub fn get_event_handlers(&self, event: &str) -> Option<&Vec<Statement>> {
|
|
59
|
+
self.events.get(event)
|
|
60
|
+
}
|
|
61
|
+
}
|
|
@@ -1,28 +1,28 @@
|
|
|
1
|
-
use std::collections::HashMap;
|
|
2
|
-
|
|
3
|
-
use
|
|
4
|
-
|
|
5
|
-
#[derive(Debug, Default, Clone, PartialEq)]
|
|
6
|
-
pub struct ImportTable {
|
|
7
|
-
pub imports: HashMap<String, Value>,
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
impl ImportTable {
|
|
11
|
-
pub fn new() -> Self {
|
|
12
|
-
ImportTable {
|
|
13
|
-
imports: HashMap::new(),
|
|
14
|
-
}
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
pub fn add_import(&mut self, name: String, value: Value) {
|
|
18
|
-
self.imports.insert(name, value);
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
pub fn get_import(&self, name: &str) -> Option<&Value> {
|
|
22
|
-
self.imports.get(name)
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
pub fn remove_import(&mut self, name: &str) -> Option<Value> {
|
|
26
|
-
self.imports.remove(name)
|
|
27
|
-
}
|
|
28
|
-
}
|
|
1
|
+
use std::collections::HashMap;
|
|
2
|
+
|
|
3
|
+
use devalang_types::Value;
|
|
4
|
+
|
|
5
|
+
#[derive(Debug, Default, Clone, PartialEq)]
|
|
6
|
+
pub struct ImportTable {
|
|
7
|
+
pub imports: HashMap<String, Value>,
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
impl ImportTable {
|
|
11
|
+
pub fn new() -> Self {
|
|
12
|
+
ImportTable {
|
|
13
|
+
imports: HashMap::new(),
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
pub fn add_import(&mut self, name: String, value: Value) {
|
|
18
|
+
self.imports.insert(name, value);
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
pub fn get_import(&self, name: &str) -> Option<&Value> {
|
|
22
|
+
self.imports.get(name)
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
pub fn remove_import(&mut self, name: &str) -> Option<Value> {
|
|
26
|
+
self.imports.remove(name)
|
|
27
|
+
}
|
|
28
|
+
}
|
package/rust/core/store/mod.rs
CHANGED
|
@@ -1,4 +1,5 @@
|
|
|
1
|
-
pub mod
|
|
2
|
-
pub mod
|
|
3
|
-
pub mod
|
|
4
|
-
pub mod
|
|
1
|
+
pub mod export;
|
|
2
|
+
pub mod function;
|
|
3
|
+
pub mod global;
|
|
4
|
+
pub mod import;
|
|
5
|
+
pub mod variable;
|
|
@@ -1,28 +1,51 @@
|
|
|
1
|
-
use std::collections::HashMap;
|
|
2
|
-
|
|
3
|
-
use
|
|
4
|
-
|
|
5
|
-
#[derive(Debug, Default, Clone, PartialEq)]
|
|
6
|
-
pub struct VariableTable {
|
|
7
|
-
pub variables: HashMap<String, Value>,
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
}
|
|
1
|
+
use std::collections::HashMap;
|
|
2
|
+
|
|
3
|
+
use devalang_types::Value;
|
|
4
|
+
|
|
5
|
+
#[derive(Debug, Default, Clone, PartialEq)]
|
|
6
|
+
pub struct VariableTable {
|
|
7
|
+
pub variables: HashMap<String, Value>,
|
|
8
|
+
pub parent: Option<Box<VariableTable>>,
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
impl VariableTable {
|
|
12
|
+
pub fn new() -> Self {
|
|
13
|
+
VariableTable {
|
|
14
|
+
variables: HashMap::new(),
|
|
15
|
+
parent: None,
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
pub fn set(&mut self, name: String, value: Value) {
|
|
20
|
+
self.variables.insert(name, value);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
pub fn with_parent(parent: VariableTable) -> Self {
|
|
24
|
+
Self {
|
|
25
|
+
variables: HashMap::new(),
|
|
26
|
+
parent: Some(Box::new(parent)),
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
pub fn get(&self, name: &str) -> Option<&Value> {
|
|
31
|
+
// First try the local table
|
|
32
|
+
if let Some(v) = self.variables.get(name) {
|
|
33
|
+
return Some(v);
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
// Then walk parent chain if present
|
|
37
|
+
let mut current = &self.parent;
|
|
38
|
+
while let Some(boxed) = current {
|
|
39
|
+
if let Some(v) = boxed.variables.get(name) {
|
|
40
|
+
return Some(v);
|
|
41
|
+
}
|
|
42
|
+
current = &boxed.parent;
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
None
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
pub fn remove(&mut self, name: &str) -> Option<Value> {
|
|
49
|
+
self.variables.remove(name)
|
|
50
|
+
}
|
|
51
|
+
}
|
package/rust/core/utils/mod.rs
CHANGED
|
@@ -1,2 +1 @@
|
|
|
1
|
-
pub mod path;
|
|
2
|
-
pub mod validation;
|
|
1
|
+
pub mod path;
|
package/rust/core/utils/path.rs
CHANGED
|
@@ -1,46 +1,37 @@
|
|
|
1
|
-
use std::path::{
|
|
2
|
-
|
|
3
|
-
pub fn find_entry_file(
|
|
4
|
-
let path = Path::new(
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
// Join the components into a normalized path
|
|
39
|
-
let normalized = components
|
|
40
|
-
.iter()
|
|
41
|
-
.map(|c| c.to_string_lossy())
|
|
42
|
-
.collect::<Vec<_>>()
|
|
43
|
-
.join("/");
|
|
44
|
-
|
|
45
|
-
format!("./{}", normalized)
|
|
46
|
-
}
|
|
1
|
+
use std::path::{Path, PathBuf};
|
|
2
|
+
|
|
3
|
+
pub fn find_entry_file(entry: &str) -> Option<String> {
|
|
4
|
+
let path = Path::new(entry);
|
|
5
|
+
|
|
6
|
+
if path.is_file() {
|
|
7
|
+
return Some(normalize_path(entry));
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
if path.is_dir() {
|
|
11
|
+
let candidate = path.join("index.deva");
|
|
12
|
+
if candidate.exists() {
|
|
13
|
+
return Some(normalize_path(&candidate));
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
None
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
pub fn normalize_path<P: AsRef<Path>>(path: P) -> String {
|
|
21
|
+
let path_buf = PathBuf::from(path.as_ref());
|
|
22
|
+
path_buf
|
|
23
|
+
.components()
|
|
24
|
+
.collect::<PathBuf>()
|
|
25
|
+
.to_string_lossy()
|
|
26
|
+
.replace('\\', "/")
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
pub fn resolve_relative_path(base: &str, import: &str) -> String {
|
|
30
|
+
let base_path = Path::new(base).parent().unwrap_or_else(|| Path::new(""));
|
|
31
|
+
let full_path = base_path.join(import);
|
|
32
|
+
full_path
|
|
33
|
+
.components()
|
|
34
|
+
.collect::<PathBuf>()
|
|
35
|
+
.to_string_lossy()
|
|
36
|
+
.replace("\\", "/")
|
|
37
|
+
}
|