@devaloop/devalang 0.0.1-beta.2 → 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/Cargo.toml +84 -81
- package/README.md +3 -2
- package/docs/CHANGELOG.md +41 -0
- package/docs/ROADMAP.md +3 -3
- package/examples/chain.deva +19 -0
- package/examples/plugin.deva +10 -10
- package/examples/routing.deva +23 -0
- package/out-tsc/bin/project-version.json +6 -0
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -8
- 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} +109 -118
- package/rust/cli/build/commands.rs +153 -153
- package/rust/cli/build/process.rs +165 -165
- package/rust/cli/check/mod.rs +208 -208
- package/rust/cli/discover/commands.rs +275 -253
- package/rust/cli/discover/config.rs +109 -111
- package/rust/cli/discover/fs.rs +19 -19
- package/rust/cli/discover/install.rs +214 -103
- package/rust/cli/discover/metadata.rs +48 -48
- package/rust/cli/discover/mod.rs +5 -5
- package/rust/cli/me/commands.rs +52 -0
- package/rust/cli/me/mod.rs +1 -0
- package/rust/cli/mod.rs +12 -12
- package/rust/cli/parser.rs +30 -69
- package/rust/cli/play/commands.rs +375 -375
- package/rust/cli/play/process.rs +159 -159
- package/rust/core/audio/engine/driver.rs +19 -2
- package/rust/core/audio/engine/export.rs +169 -169
- package/rust/core/audio/engine/mod.rs +56 -56
- package/rust/core/audio/engine/notes/dsp.rs +88 -85
- package/rust/core/audio/engine/notes/mod.rs +53 -44
- package/rust/core/audio/engine/notes/params.rs +294 -294
- package/rust/core/audio/engine/sample/insert.rs +148 -47
- package/rust/core/audio/engine/sample/mod.rs +40 -40
- package/rust/core/audio/engine/sample/padding.rs +170 -170
- package/rust/core/audio/evaluator/condition.rs +61 -61
- package/rust/core/audio/evaluator/numeric.rs +152 -152
- package/rust/core/audio/evaluator/rhs.rs +16 -16
- package/rust/core/audio/evaluator/string_expr.rs +94 -94
- package/rust/core/audio/interpreter/driver.rs +574 -574
- package/rust/core/audio/interpreter/mod.rs +2 -2
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +9 -5
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -384
- package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +1 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +66 -11
- package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -3
- package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -192
- package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -24
- package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -116
- package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -97
- package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -100
- package/rust/core/audio/interpreter/statements/automate.rs +16 -16
- package/rust/core/audio/interpreter/statements/call.rs +31 -1
- package/rust/core/audio/interpreter/statements/condition.rs +72 -72
- package/rust/core/audio/interpreter/statements/function.rs +24 -24
- package/rust/core/audio/interpreter/statements/let_.rs +36 -36
- package/rust/core/audio/interpreter/statements/load.rs +17 -17
- package/rust/core/audio/interpreter/statements/loop_.rs +115 -115
- package/rust/core/audio/interpreter/statements/spawn.rs +51 -2
- package/rust/core/audio/interpreter/statements/trigger.rs +242 -239
- package/rust/core/audio/loader/trigger.rs +98 -98
- package/rust/core/audio/player.rs +70 -70
- package/rust/core/audio/special/mod.rs +9 -9
- package/rust/core/builder/mod.rs +129 -129
- package/rust/core/debugger/lexer.rs +27 -27
- package/rust/core/debugger/logs.rs +52 -52
- package/rust/core/debugger/preprocessor.rs +27 -27
- package/rust/core/debugger/store.rs +38 -38
- package/rust/core/lexer/driver.rs +59 -59
- package/rust/core/lexer/handler/arrow.rs +82 -82
- package/rust/core/lexer/handler/at.rs +21 -21
- package/rust/core/lexer/handler/brace.rs +41 -41
- package/rust/core/lexer/handler/colon.rs +21 -21
- package/rust/core/lexer/handler/comment.rs +30 -30
- package/rust/core/lexer/handler/dot.rs +21 -21
- package/rust/core/lexer/handler/driver.rs +337 -337
- package/rust/core/lexer/handler/identifier.rs +47 -47
- package/rust/core/lexer/handler/indent.rs +66 -66
- package/rust/core/lexer/handler/mod.rs +15 -15
- package/rust/core/lexer/handler/newline.rs +23 -23
- package/rust/core/lexer/handler/number.rs +31 -31
- package/rust/core/lexer/handler/operator.rs +46 -46
- package/rust/core/lexer/handler/parenthesis.rs +41 -41
- package/rust/core/lexer/handler/slash.rs +21 -21
- package/rust/core/lexer/handler/string.rs +63 -63
- package/rust/core/lexer/mod.rs +3 -3
- package/rust/core/mod.rs +9 -9
- package/rust/core/parser/driver/block.rs +111 -111
- package/rust/core/parser/driver/cursor.rs +82 -82
- package/rust/core/parser/driver/driver_impl.rs +21 -1
- package/rust/core/parser/driver/mod.rs +6 -6
- package/rust/core/parser/driver/parse_array.rs +120 -120
- package/rust/core/parser/driver/parse_map.rs +247 -223
- package/rust/core/parser/driver/parser.rs +160 -160
- package/rust/core/parser/handler/arrow_call.rs +65 -14
- package/rust/core/parser/handler/identifier/synth.rs +171 -135
- package/rust/core/parser/handler/mod.rs +9 -9
- package/rust/core/parser/handler/pattern.rs +24 -1
- package/rust/core/plugin/loader.rs +137 -137
- package/rust/core/plugin/mod.rs +2 -2
- package/rust/core/plugin/runner/non_wasm.rs +481 -297
- package/rust/core/plugin/runner/wasm32.rs +1 -0
- package/rust/core/preprocessor/loader/inject.rs +313 -278
- package/rust/core/preprocessor/loader/loader_helpers.rs +110 -110
- package/rust/core/preprocessor/loader/mod.rs +235 -235
- package/rust/core/preprocessor/module.rs +55 -55
- package/rust/core/preprocessor/processor/handlers.rs +107 -107
- package/rust/core/preprocessor/resolver/bank.rs +49 -49
- package/rust/core/preprocessor/resolver/call.rs +124 -124
- package/rust/core/preprocessor/resolver/condition.rs +95 -95
- package/rust/core/preprocessor/resolver/driver.rs +324 -324
- package/rust/core/preprocessor/resolver/function.rs +69 -69
- package/rust/core/preprocessor/resolver/group.rs +122 -122
- package/rust/core/preprocessor/resolver/let_.rs +32 -32
- package/rust/core/preprocessor/resolver/loop_.rs +318 -318
- package/rust/core/preprocessor/resolver/mod.rs +16 -16
- package/rust/core/preprocessor/resolver/pattern.rs +95 -83
- package/rust/core/preprocessor/resolver/spawn.rs +99 -99
- package/rust/core/preprocessor/resolver/synth.rs +54 -54
- package/rust/core/preprocessor/resolver/tempo.rs +48 -48
- package/rust/core/preprocessor/resolver/trigger.rs +116 -116
- package/rust/core/preprocessor/resolver/value.rs +176 -176
- package/rust/core/store/global.rs +57 -57
- package/rust/lib.rs +323 -323
- package/rust/macros/Cargo.toml +14 -0
- package/rust/macros/src/lib.rs +52 -0
- package/rust/main.rs +311 -142
- package/rust/types/Cargo.toml +1 -1
- package/rust/types/src/addons.rs +3 -1
- package/rust/types/src/config.rs +1 -3
- package/rust/utils/Cargo.toml +5 -2
- package/rust/utils/src/file.rs +397 -14
- package/rust/utils/src/path.rs +31 -2
- package/rust/utils/src/version.rs +38 -7
- package/rust/web/auth.rs +5 -0
- package/rust/web/forge.rs +5 -0
- package/rust/web/mod.rs +5 -3
- 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 -306
- package/rust/cli/bank/mod.rs +0 -29
- package/rust/cli/install/bank.rs +0 -72
- package/rust/cli/install/commands.rs +0 -35
- package/rust/cli/install/mod.rs +0 -4
- package/rust/cli/install/plugin.rs +0 -80
|
@@ -1,235 +1,235 @@
|
|
|
1
|
-
#[cfg(feature = "cli")]
|
|
2
|
-
use crate::core::preprocessor::resolver::driver::{
|
|
3
|
-
resolve_all_modules, resolve_and_flatten_all_modules,
|
|
4
|
-
};
|
|
5
|
-
// resolve_relative_path moved to loader_helpers
|
|
6
|
-
use crate::core::{
|
|
7
|
-
error::ErrorHandler,
|
|
8
|
-
lexer::{driver::Lexer, token::Token},
|
|
9
|
-
parser::{driver::parser::Parser, statement::Statement},
|
|
10
|
-
preprocessor::{module::Module, processor::handlers::process_modules},
|
|
11
|
-
store::global::GlobalStore,
|
|
12
|
-
};
|
|
13
|
-
use devalang_utils::path::normalize_path;
|
|
14
|
-
use std::{collections::HashMap, path::Path};
|
|
15
|
-
|
|
16
|
-
mod inject;
|
|
17
|
-
mod loader_helpers;
|
|
18
|
-
|
|
19
|
-
pub struct ModuleLoader {
|
|
20
|
-
pub entry: String,
|
|
21
|
-
pub output: String,
|
|
22
|
-
pub base_dir: String,
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
impl ModuleLoader {
|
|
26
|
-
pub fn new(entry: &str, output: &str) -> Self {
|
|
27
|
-
let base_dir = Path::new(entry)
|
|
28
|
-
.parent()
|
|
29
|
-
.unwrap_or(Path::new(""))
|
|
30
|
-
.to_string_lossy()
|
|
31
|
-
.replace('\\', "/");
|
|
32
|
-
|
|
33
|
-
Self {
|
|
34
|
-
entry: entry.to_string(),
|
|
35
|
-
output: output.to_string(),
|
|
36
|
-
base_dir,
|
|
37
|
-
}
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
pub fn from_raw_source(
|
|
41
|
-
entry_path: &str,
|
|
42
|
-
output_path: &str,
|
|
43
|
-
content: &str,
|
|
44
|
-
global_store: &mut GlobalStore,
|
|
45
|
-
) -> Self {
|
|
46
|
-
let normalized_entry_path = normalize_path(entry_path);
|
|
47
|
-
|
|
48
|
-
let mut module = Module::new(entry_path);
|
|
49
|
-
module.content = content.to_string();
|
|
50
|
-
|
|
51
|
-
// Insert a module stub containing the provided content into the
|
|
52
|
-
// global store. This is used by the WASM APIs and tests which
|
|
53
|
-
// operate on in-memory sources instead of files on disk.
|
|
54
|
-
global_store.insert_module(normalized_entry_path.to_string(), module);
|
|
55
|
-
|
|
56
|
-
Self {
|
|
57
|
-
entry: normalized_entry_path.to_string(),
|
|
58
|
-
output: output_path.to_string(),
|
|
59
|
-
base_dir: "".to_string(),
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
pub fn extract_statements_map(
|
|
64
|
-
&self,
|
|
65
|
-
global_store: &GlobalStore,
|
|
66
|
-
) -> HashMap<String, Vec<Statement>> {
|
|
67
|
-
global_store
|
|
68
|
-
.modules
|
|
69
|
-
.iter()
|
|
70
|
-
.map(|(path, module)| (path.clone(), module.statements.clone()))
|
|
71
|
-
.collect()
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
pub fn load_single_module(&self, global_store: &mut GlobalStore) -> Result<Module, String> {
|
|
75
|
-
let mut module = global_store
|
|
76
|
-
.modules
|
|
77
|
-
.remove(&self.entry)
|
|
78
|
-
.ok_or_else(|| format!("Module not found in store for path: {}", self.entry))?;
|
|
79
|
-
|
|
80
|
-
// SECTION Lexing the module content
|
|
81
|
-
let lexer = Lexer::new();
|
|
82
|
-
let tokens = lexer
|
|
83
|
-
.lex_from_source(&module.content)
|
|
84
|
-
.map_err(|e| format!("Lexer failed: {}", e))?;
|
|
85
|
-
|
|
86
|
-
module.tokens = tokens.clone();
|
|
87
|
-
|
|
88
|
-
// SECTION Parsing tokens into statements
|
|
89
|
-
let mut parser = Parser::new();
|
|
90
|
-
parser.set_current_module(self.entry.clone());
|
|
91
|
-
let statements = parser.parse_tokens(tokens, global_store);
|
|
92
|
-
module.statements = statements;
|
|
93
|
-
|
|
94
|
-
// SECTION Injecting bank triggers if any (legacy default for single-module run)
|
|
95
|
-
if let Err(e) = inject::inject_bank_triggers(&mut module, "808", None) {
|
|
96
|
-
return Err(format!("Failed to inject bank triggers: {}", e));
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
for (plugin_name, alias) in inject::extract_plugin_uses(&module.statements) {
|
|
100
|
-
inject::load_plugin_and_register(&mut module, &plugin_name, &alias, global_store);
|
|
101
|
-
}
|
|
102
|
-
|
|
103
|
-
global_store
|
|
104
|
-
.modules
|
|
105
|
-
.insert(self.entry.clone(), module.clone());
|
|
106
|
-
|
|
107
|
-
// SECTION Error handling
|
|
108
|
-
let mut error_handler = ErrorHandler::new();
|
|
109
|
-
error_handler.detect_from_statements(&mut parser, &module.statements);
|
|
110
|
-
|
|
111
|
-
Ok(module)
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
pub fn load_wasm_module(&self, global_store: &mut GlobalStore) -> Result<(), String> {
|
|
115
|
-
// Step one : Load the module from the global store
|
|
116
|
-
let module = {
|
|
117
|
-
let module_ref = global_store
|
|
118
|
-
.modules
|
|
119
|
-
.get(&self.entry)
|
|
120
|
-
.ok_or_else(|| format!("❌ Module not found for path: {}", self.entry))?;
|
|
121
|
-
|
|
122
|
-
Module::from_existing(&self.entry, module_ref.content.clone())
|
|
123
|
-
};
|
|
124
|
-
|
|
125
|
-
// Step two : lexing
|
|
126
|
-
let lexer = Lexer::new();
|
|
127
|
-
let tokens = lexer
|
|
128
|
-
.lex_from_source(&module.content)
|
|
129
|
-
.map_err(|e| format!("Lexer failed: {}", e))?;
|
|
130
|
-
|
|
131
|
-
// Step three : parsing
|
|
132
|
-
let mut parser = Parser::new();
|
|
133
|
-
parser.set_current_module(self.entry.clone());
|
|
134
|
-
|
|
135
|
-
let statements = parser.parse_tokens(tokens.clone(), global_store);
|
|
136
|
-
|
|
137
|
-
let mut updated_module = module;
|
|
138
|
-
updated_module.tokens = tokens;
|
|
139
|
-
updated_module.statements = statements;
|
|
140
|
-
|
|
141
|
-
// Step four : Injecting bank triggers if any
|
|
142
|
-
if let Err(e) = inject::inject_bank_triggers(&mut updated_module, "808", None) {
|
|
143
|
-
return Err(format!("Failed to inject bank triggers: {}", e));
|
|
144
|
-
}
|
|
145
|
-
|
|
146
|
-
// Insert the updated module into the global store before processing so
|
|
147
|
-
// process_modules can operate on it and populate variable_table, imports,
|
|
148
|
-
// and other derived structures.
|
|
149
|
-
global_store
|
|
150
|
-
.modules
|
|
151
|
-
.insert(self.entry.clone(), updated_module.clone());
|
|
152
|
-
|
|
153
|
-
// Process modules to populate module.variable_table, import/export tables,
|
|
154
|
-
// and other derived structures so runtime execution can resolve groups/synths.
|
|
155
|
-
process_modules(self, global_store);
|
|
156
|
-
|
|
157
|
-
for (plugin_name, alias) in inject::extract_plugin_uses(&updated_module.statements) {
|
|
158
|
-
inject::load_plugin_and_register(
|
|
159
|
-
&mut updated_module,
|
|
160
|
-
&plugin_name,
|
|
161
|
-
&alias,
|
|
162
|
-
global_store,
|
|
163
|
-
);
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
// Step four : error handling
|
|
167
|
-
let mut error_handler = ErrorHandler::new();
|
|
168
|
-
error_handler.detect_from_statements(&mut parser, &updated_module.statements);
|
|
169
|
-
|
|
170
|
-
// Final step : also expose module-level variables and functions into the global store
|
|
171
|
-
// so runtime evaluation (render_audio) can find group/synth definitions.
|
|
172
|
-
// Use the module instance that was actually processed by `process_modules`
|
|
173
|
-
// (it lives in `global_store.modules`) because `updated_module` is a local
|
|
174
|
-
// clone and won't contain the mutations applied by `process_modules`.
|
|
175
|
-
if let Some(stored_module) = global_store.modules.get(&self.entry) {
|
|
176
|
-
global_store
|
|
177
|
-
.variables
|
|
178
|
-
.variables
|
|
179
|
-
.extend(stored_module.variable_table.variables.clone());
|
|
180
|
-
global_store
|
|
181
|
-
.functions
|
|
182
|
-
.functions
|
|
183
|
-
.extend(stored_module.function_table.functions.clone());
|
|
184
|
-
} else {
|
|
185
|
-
// Fallback to the local updated_module if for any reason the module
|
|
186
|
-
// wasn't inserted into the store (defensive programming).
|
|
187
|
-
global_store
|
|
188
|
-
.variables
|
|
189
|
-
.variables
|
|
190
|
-
.extend(updated_module.variable_table.variables.clone());
|
|
191
|
-
global_store
|
|
192
|
-
.functions
|
|
193
|
-
.functions
|
|
194
|
-
.extend(updated_module.function_table.functions.clone());
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
Ok(())
|
|
198
|
-
}
|
|
199
|
-
|
|
200
|
-
#[cfg(feature = "cli")]
|
|
201
|
-
pub fn load_all_modules(
|
|
202
|
-
&self,
|
|
203
|
-
global_store: &mut GlobalStore,
|
|
204
|
-
) -> (HashMap<String, Vec<Token>>, HashMap<String, Vec<Statement>>) {
|
|
205
|
-
// SECTION Load the entry module and its dependencies
|
|
206
|
-
let tokens_by_module = self.load_module_recursively(&self.entry, global_store);
|
|
207
|
-
|
|
208
|
-
// SECTION Process and resolve modules
|
|
209
|
-
process_modules(self, global_store);
|
|
210
|
-
resolve_all_modules(self, global_store);
|
|
211
|
-
|
|
212
|
-
// SECTION Flatten all modules to get statements (+ injects)
|
|
213
|
-
let statements_by_module = resolve_and_flatten_all_modules(global_store);
|
|
214
|
-
|
|
215
|
-
(tokens_by_module, statements_by_module)
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
#[cfg(feature = "cli")]
|
|
219
|
-
fn load_module_recursively(
|
|
220
|
-
&self,
|
|
221
|
-
raw_path: &str,
|
|
222
|
-
global_store: &mut GlobalStore,
|
|
223
|
-
) -> HashMap<String, Vec<Token>> {
|
|
224
|
-
crate::core::preprocessor::loader::loader_helpers::load_module_recursively(
|
|
225
|
-
raw_path,
|
|
226
|
-
global_store,
|
|
227
|
-
)
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
#[cfg(feature = "cli")]
|
|
231
|
-
#[allow(dead_code)]
|
|
232
|
-
fn load_module_imports(&self, path: &String, global_store: &mut GlobalStore) {
|
|
233
|
-
crate::core::preprocessor::loader::loader_helpers::load_module_imports(path, global_store)
|
|
234
|
-
}
|
|
235
|
-
}
|
|
1
|
+
#[cfg(feature = "cli")]
|
|
2
|
+
use crate::core::preprocessor::resolver::driver::{
|
|
3
|
+
resolve_all_modules, resolve_and_flatten_all_modules,
|
|
4
|
+
};
|
|
5
|
+
// resolve_relative_path moved to loader_helpers
|
|
6
|
+
use crate::core::{
|
|
7
|
+
error::ErrorHandler,
|
|
8
|
+
lexer::{driver::Lexer, token::Token},
|
|
9
|
+
parser::{driver::parser::Parser, statement::Statement},
|
|
10
|
+
preprocessor::{module::Module, processor::handlers::process_modules},
|
|
11
|
+
store::global::GlobalStore,
|
|
12
|
+
};
|
|
13
|
+
use devalang_utils::path::normalize_path;
|
|
14
|
+
use std::{collections::HashMap, path::Path};
|
|
15
|
+
|
|
16
|
+
mod inject;
|
|
17
|
+
mod loader_helpers;
|
|
18
|
+
|
|
19
|
+
pub struct ModuleLoader {
|
|
20
|
+
pub entry: String,
|
|
21
|
+
pub output: String,
|
|
22
|
+
pub base_dir: String,
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
impl ModuleLoader {
|
|
26
|
+
pub fn new(entry: &str, output: &str) -> Self {
|
|
27
|
+
let base_dir = Path::new(entry)
|
|
28
|
+
.parent()
|
|
29
|
+
.unwrap_or(Path::new(""))
|
|
30
|
+
.to_string_lossy()
|
|
31
|
+
.replace('\\', "/");
|
|
32
|
+
|
|
33
|
+
Self {
|
|
34
|
+
entry: entry.to_string(),
|
|
35
|
+
output: output.to_string(),
|
|
36
|
+
base_dir,
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
pub fn from_raw_source(
|
|
41
|
+
entry_path: &str,
|
|
42
|
+
output_path: &str,
|
|
43
|
+
content: &str,
|
|
44
|
+
global_store: &mut GlobalStore,
|
|
45
|
+
) -> Self {
|
|
46
|
+
let normalized_entry_path = normalize_path(entry_path);
|
|
47
|
+
|
|
48
|
+
let mut module = Module::new(entry_path);
|
|
49
|
+
module.content = content.to_string();
|
|
50
|
+
|
|
51
|
+
// Insert a module stub containing the provided content into the
|
|
52
|
+
// global store. This is used by the WASM APIs and tests which
|
|
53
|
+
// operate on in-memory sources instead of files on disk.
|
|
54
|
+
global_store.insert_module(normalized_entry_path.to_string(), module);
|
|
55
|
+
|
|
56
|
+
Self {
|
|
57
|
+
entry: normalized_entry_path.to_string(),
|
|
58
|
+
output: output_path.to_string(),
|
|
59
|
+
base_dir: "".to_string(),
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
pub fn extract_statements_map(
|
|
64
|
+
&self,
|
|
65
|
+
global_store: &GlobalStore,
|
|
66
|
+
) -> HashMap<String, Vec<Statement>> {
|
|
67
|
+
global_store
|
|
68
|
+
.modules
|
|
69
|
+
.iter()
|
|
70
|
+
.map(|(path, module)| (path.clone(), module.statements.clone()))
|
|
71
|
+
.collect()
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
pub fn load_single_module(&self, global_store: &mut GlobalStore) -> Result<Module, String> {
|
|
75
|
+
let mut module = global_store
|
|
76
|
+
.modules
|
|
77
|
+
.remove(&self.entry)
|
|
78
|
+
.ok_or_else(|| format!("Module not found in store for path: {}", self.entry))?;
|
|
79
|
+
|
|
80
|
+
// SECTION Lexing the module content
|
|
81
|
+
let lexer = Lexer::new();
|
|
82
|
+
let tokens = lexer
|
|
83
|
+
.lex_from_source(&module.content)
|
|
84
|
+
.map_err(|e| format!("Lexer failed: {}", e))?;
|
|
85
|
+
|
|
86
|
+
module.tokens = tokens.clone();
|
|
87
|
+
|
|
88
|
+
// SECTION Parsing tokens into statements
|
|
89
|
+
let mut parser = Parser::new();
|
|
90
|
+
parser.set_current_module(self.entry.clone());
|
|
91
|
+
let statements = parser.parse_tokens(tokens, global_store);
|
|
92
|
+
module.statements = statements;
|
|
93
|
+
|
|
94
|
+
// SECTION Injecting bank triggers if any (legacy default for single-module run)
|
|
95
|
+
if let Err(e) = inject::inject_bank_triggers(&mut module, "808", None) {
|
|
96
|
+
return Err(format!("Failed to inject bank triggers: {}", e));
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
for (plugin_name, alias) in inject::extract_plugin_uses(&module.statements) {
|
|
100
|
+
inject::load_plugin_and_register(&mut module, &plugin_name, &alias, global_store);
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
global_store
|
|
104
|
+
.modules
|
|
105
|
+
.insert(self.entry.clone(), module.clone());
|
|
106
|
+
|
|
107
|
+
// SECTION Error handling
|
|
108
|
+
let mut error_handler = ErrorHandler::new();
|
|
109
|
+
error_handler.detect_from_statements(&mut parser, &module.statements);
|
|
110
|
+
|
|
111
|
+
Ok(module)
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
pub fn load_wasm_module(&self, global_store: &mut GlobalStore) -> Result<(), String> {
|
|
115
|
+
// Step one : Load the module from the global store
|
|
116
|
+
let module = {
|
|
117
|
+
let module_ref = global_store
|
|
118
|
+
.modules
|
|
119
|
+
.get(&self.entry)
|
|
120
|
+
.ok_or_else(|| format!("❌ Module not found for path: {}", self.entry))?;
|
|
121
|
+
|
|
122
|
+
Module::from_existing(&self.entry, module_ref.content.clone())
|
|
123
|
+
};
|
|
124
|
+
|
|
125
|
+
// Step two : lexing
|
|
126
|
+
let lexer = Lexer::new();
|
|
127
|
+
let tokens = lexer
|
|
128
|
+
.lex_from_source(&module.content)
|
|
129
|
+
.map_err(|e| format!("Lexer failed: {}", e))?;
|
|
130
|
+
|
|
131
|
+
// Step three : parsing
|
|
132
|
+
let mut parser = Parser::new();
|
|
133
|
+
parser.set_current_module(self.entry.clone());
|
|
134
|
+
|
|
135
|
+
let statements = parser.parse_tokens(tokens.clone(), global_store);
|
|
136
|
+
|
|
137
|
+
let mut updated_module = module;
|
|
138
|
+
updated_module.tokens = tokens;
|
|
139
|
+
updated_module.statements = statements;
|
|
140
|
+
|
|
141
|
+
// Step four : Injecting bank triggers if any
|
|
142
|
+
if let Err(e) = inject::inject_bank_triggers(&mut updated_module, "808", None) {
|
|
143
|
+
return Err(format!("Failed to inject bank triggers: {}", e));
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
// Insert the updated module into the global store before processing so
|
|
147
|
+
// process_modules can operate on it and populate variable_table, imports,
|
|
148
|
+
// and other derived structures.
|
|
149
|
+
global_store
|
|
150
|
+
.modules
|
|
151
|
+
.insert(self.entry.clone(), updated_module.clone());
|
|
152
|
+
|
|
153
|
+
// Process modules to populate module.variable_table, import/export tables,
|
|
154
|
+
// and other derived structures so runtime execution can resolve groups/synths.
|
|
155
|
+
process_modules(self, global_store);
|
|
156
|
+
|
|
157
|
+
for (plugin_name, alias) in inject::extract_plugin_uses(&updated_module.statements) {
|
|
158
|
+
inject::load_plugin_and_register(
|
|
159
|
+
&mut updated_module,
|
|
160
|
+
&plugin_name,
|
|
161
|
+
&alias,
|
|
162
|
+
global_store,
|
|
163
|
+
);
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
// Step four : error handling
|
|
167
|
+
let mut error_handler = ErrorHandler::new();
|
|
168
|
+
error_handler.detect_from_statements(&mut parser, &updated_module.statements);
|
|
169
|
+
|
|
170
|
+
// Final step : also expose module-level variables and functions into the global store
|
|
171
|
+
// so runtime evaluation (render_audio) can find group/synth definitions.
|
|
172
|
+
// Use the module instance that was actually processed by `process_modules`
|
|
173
|
+
// (it lives in `global_store.modules`) because `updated_module` is a local
|
|
174
|
+
// clone and won't contain the mutations applied by `process_modules`.
|
|
175
|
+
if let Some(stored_module) = global_store.modules.get(&self.entry) {
|
|
176
|
+
global_store
|
|
177
|
+
.variables
|
|
178
|
+
.variables
|
|
179
|
+
.extend(stored_module.variable_table.variables.clone());
|
|
180
|
+
global_store
|
|
181
|
+
.functions
|
|
182
|
+
.functions
|
|
183
|
+
.extend(stored_module.function_table.functions.clone());
|
|
184
|
+
} else {
|
|
185
|
+
// Fallback to the local updated_module if for any reason the module
|
|
186
|
+
// wasn't inserted into the store (defensive programming).
|
|
187
|
+
global_store
|
|
188
|
+
.variables
|
|
189
|
+
.variables
|
|
190
|
+
.extend(updated_module.variable_table.variables.clone());
|
|
191
|
+
global_store
|
|
192
|
+
.functions
|
|
193
|
+
.functions
|
|
194
|
+
.extend(updated_module.function_table.functions.clone());
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
Ok(())
|
|
198
|
+
}
|
|
199
|
+
|
|
200
|
+
#[cfg(feature = "cli")]
|
|
201
|
+
pub fn load_all_modules(
|
|
202
|
+
&self,
|
|
203
|
+
global_store: &mut GlobalStore,
|
|
204
|
+
) -> (HashMap<String, Vec<Token>>, HashMap<String, Vec<Statement>>) {
|
|
205
|
+
// SECTION Load the entry module and its dependencies
|
|
206
|
+
let tokens_by_module = self.load_module_recursively(&self.entry, global_store);
|
|
207
|
+
|
|
208
|
+
// SECTION Process and resolve modules
|
|
209
|
+
process_modules(self, global_store);
|
|
210
|
+
resolve_all_modules(self, global_store);
|
|
211
|
+
|
|
212
|
+
// SECTION Flatten all modules to get statements (+ injects)
|
|
213
|
+
let statements_by_module = resolve_and_flatten_all_modules(global_store);
|
|
214
|
+
|
|
215
|
+
(tokens_by_module, statements_by_module)
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
#[cfg(feature = "cli")]
|
|
219
|
+
fn load_module_recursively(
|
|
220
|
+
&self,
|
|
221
|
+
raw_path: &str,
|
|
222
|
+
global_store: &mut GlobalStore,
|
|
223
|
+
) -> HashMap<String, Vec<Token>> {
|
|
224
|
+
crate::core::preprocessor::loader::loader_helpers::load_module_recursively(
|
|
225
|
+
raw_path,
|
|
226
|
+
global_store,
|
|
227
|
+
)
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
#[cfg(feature = "cli")]
|
|
231
|
+
#[allow(dead_code)]
|
|
232
|
+
fn load_module_imports(&self, path: &String, global_store: &mut GlobalStore) {
|
|
233
|
+
crate::core::preprocessor::loader::loader_helpers::load_module_imports(path, global_store)
|
|
234
|
+
}
|
|
235
|
+
}
|
|
@@ -1,55 +1,55 @@
|
|
|
1
|
-
use crate::core::{lexer::token::Token, parser::statement::Statement};
|
|
2
|
-
use devalang_types::{ExportTable, FunctionTable, ImportTable, VariableTable};
|
|
3
|
-
|
|
4
|
-
#[derive(Debug, Clone)]
|
|
5
|
-
pub struct Module {
|
|
6
|
-
pub path: String,
|
|
7
|
-
pub resolved: bool,
|
|
8
|
-
pub tokens: Vec<Token>,
|
|
9
|
-
pub statements: Vec<Statement>,
|
|
10
|
-
pub variable_table: VariableTable,
|
|
11
|
-
pub function_table: FunctionTable,
|
|
12
|
-
pub export_table: ExportTable,
|
|
13
|
-
pub import_table: ImportTable,
|
|
14
|
-
pub content: String,
|
|
15
|
-
pub current_dir: String,
|
|
16
|
-
}
|
|
17
|
-
|
|
18
|
-
impl Module {
|
|
19
|
-
pub fn new(path: &str) -> Self {
|
|
20
|
-
Module {
|
|
21
|
-
path: path.to_string(),
|
|
22
|
-
tokens: Vec::new(),
|
|
23
|
-
statements: Vec::new(),
|
|
24
|
-
variable_table: VariableTable::new(),
|
|
25
|
-
function_table: FunctionTable::new(),
|
|
26
|
-
export_table: ExportTable::new(),
|
|
27
|
-
import_table: ImportTable::new(),
|
|
28
|
-
resolved: false,
|
|
29
|
-
content: String::new(),
|
|
30
|
-
current_dir: String::new(),
|
|
31
|
-
}
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
pub fn is_resolved(&self) -> bool {
|
|
35
|
-
self.resolved
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
pub fn set_resolved(&mut self, resolved: bool) {
|
|
39
|
-
self.resolved = resolved;
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
pub fn add_token(&mut self, token: Token) {
|
|
43
|
-
self.tokens.push(token);
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
pub fn add_statement(&mut self, statement: Statement) {
|
|
47
|
-
self.statements.push(statement);
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
pub fn from_existing(path: &str, content: String) -> Self {
|
|
51
|
-
let mut module = Module::new(path);
|
|
52
|
-
module.content = content;
|
|
53
|
-
module
|
|
54
|
-
}
|
|
55
|
-
}
|
|
1
|
+
use crate::core::{lexer::token::Token, parser::statement::Statement};
|
|
2
|
+
use devalang_types::{ExportTable, FunctionTable, ImportTable, VariableTable};
|
|
3
|
+
|
|
4
|
+
#[derive(Debug, Clone)]
|
|
5
|
+
pub struct Module {
|
|
6
|
+
pub path: String,
|
|
7
|
+
pub resolved: bool,
|
|
8
|
+
pub tokens: Vec<Token>,
|
|
9
|
+
pub statements: Vec<Statement>,
|
|
10
|
+
pub variable_table: VariableTable,
|
|
11
|
+
pub function_table: FunctionTable,
|
|
12
|
+
pub export_table: ExportTable,
|
|
13
|
+
pub import_table: ImportTable,
|
|
14
|
+
pub content: String,
|
|
15
|
+
pub current_dir: String,
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
impl Module {
|
|
19
|
+
pub fn new(path: &str) -> Self {
|
|
20
|
+
Module {
|
|
21
|
+
path: path.to_string(),
|
|
22
|
+
tokens: Vec::new(),
|
|
23
|
+
statements: Vec::new(),
|
|
24
|
+
variable_table: VariableTable::new(),
|
|
25
|
+
function_table: FunctionTable::new(),
|
|
26
|
+
export_table: ExportTable::new(),
|
|
27
|
+
import_table: ImportTable::new(),
|
|
28
|
+
resolved: false,
|
|
29
|
+
content: String::new(),
|
|
30
|
+
current_dir: String::new(),
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
pub fn is_resolved(&self) -> bool {
|
|
35
|
+
self.resolved
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
pub fn set_resolved(&mut self, resolved: bool) {
|
|
39
|
+
self.resolved = resolved;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
pub fn add_token(&mut self, token: Token) {
|
|
43
|
+
self.tokens.push(token);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
pub fn add_statement(&mut self, statement: Statement) {
|
|
47
|
+
self.statements.push(statement);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
pub fn from_existing(path: &str, content: String) -> Self {
|
|
51
|
+
let mut module = Module::new(path);
|
|
52
|
+
module.content = content;
|
|
53
|
+
module
|
|
54
|
+
}
|
|
55
|
+
}
|