@devaloop/devalang 0.0.1-beta.1 → 0.0.1-beta.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (220) hide show
  1. package/.devalang +9 -10
  2. package/Cargo.toml +5 -4
  3. package/README.md +7 -5
  4. package/docs/CHANGELOG.md +42 -0
  5. package/docs/ROADMAP.md +5 -1
  6. package/docs/TODO.md +3 -14
  7. package/examples/bus.deva +10 -0
  8. package/examples/effect.deva +2 -0
  9. package/examples/filter.deva +11 -0
  10. package/examples/lfo.deva +9 -0
  11. package/examples/synth.deva +11 -1
  12. package/examples/synth_types.deva +17 -0
  13. package/out-tsc/core/functions/index.d.ts +5 -0
  14. package/out-tsc/core/functions/index.js +11 -0
  15. package/out-tsc/pkg/devalang_core.d.ts +2 -0
  16. package/out-tsc/pkg/devalang_core.js +17 -2
  17. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +8 -7
  18. package/package.json +1 -1
  19. package/project-version.json +3 -3
  20. package/rust/cli/bank/api.rs +122 -122
  21. package/rust/cli/bank/commands.rs +33 -2
  22. package/rust/cli/bank/mod.rs +29 -29
  23. package/rust/cli/build/commands.rs +53 -3
  24. package/rust/cli/build/mod.rs +2 -2
  25. package/rust/cli/build/process.rs +26 -7
  26. package/rust/cli/check/mod.rs +2 -2
  27. package/rust/cli/discover/commands.rs +253 -253
  28. package/rust/cli/discover/config.rs +111 -111
  29. package/rust/cli/discover/fs.rs +19 -19
  30. package/rust/cli/discover/install.rs +103 -103
  31. package/rust/cli/discover/metadata.rs +48 -48
  32. package/rust/cli/discover/mod.rs +5 -5
  33. package/rust/cli/install/addon.rs +118 -118
  34. package/rust/cli/install/bank.rs +22 -3
  35. package/rust/cli/install/commands.rs +35 -35
  36. package/rust/cli/install/mod.rs +4 -4
  37. package/rust/cli/install/plugin.rs +80 -61
  38. package/rust/cli/login/commands.rs +124 -124
  39. package/rust/cli/mod.rs +12 -12
  40. package/rust/cli/parser.rs +46 -1
  41. package/rust/cli/play/commands.rs +71 -20
  42. package/rust/cli/play/mod.rs +5 -5
  43. package/rust/cli/play/process.rs +14 -5
  44. package/rust/cli/play/realtime.rs +91 -91
  45. package/rust/cli/telemetry/commands.rs +22 -22
  46. package/rust/cli/telemetry/event_creator.rs +80 -80
  47. package/rust/cli/telemetry/mod.rs +3 -3
  48. package/rust/cli/telemetry/send.rs +51 -51
  49. package/rust/cli/template/commands.rs +69 -69
  50. package/rust/config/driver.rs +112 -103
  51. package/rust/config/mod.rs +3 -3
  52. package/rust/config/ops.rs +26 -26
  53. package/rust/config/settings.rs +101 -101
  54. package/rust/core/audio/engine/driver.rs +220 -0
  55. package/rust/core/audio/engine/export.rs +169 -0
  56. package/rust/core/audio/engine/helpers.rs +178 -170
  57. package/rust/core/audio/engine/mod.rs +51 -2
  58. package/rust/core/audio/engine/notes/dsp.rs +85 -0
  59. package/rust/core/audio/engine/notes/mod.rs +44 -0
  60. package/rust/core/audio/engine/notes/params.rs +294 -0
  61. package/rust/core/audio/engine/sample/insert.rs +199 -0
  62. package/rust/core/audio/engine/sample/mod.rs +40 -0
  63. package/rust/core/audio/engine/sample/padding.rs +170 -0
  64. package/rust/core/audio/evaluator/condition.rs +61 -0
  65. package/rust/core/audio/evaluator/mod.rs +9 -0
  66. package/rust/core/audio/{evaluator.rs → evaluator/numeric.rs} +1 -159
  67. package/rust/core/audio/evaluator/rhs.rs +16 -0
  68. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  69. package/rust/core/audio/interpreter/driver.rs +55 -23
  70. package/rust/core/audio/interpreter/mod.rs +1 -13
  71. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
  72. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
  73. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
  74. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
  75. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  76. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  77. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  78. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  79. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  80. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  81. package/rust/core/audio/interpreter/{automate.rs → statements/automate.rs} +16 -18
  82. package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +5 -4
  83. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +2 -1
  84. package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +2 -4
  85. package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +2 -4
  86. package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +2 -4
  87. package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +2 -1
  88. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  89. package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
  90. package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +3 -2
  91. package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
  92. package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +1 -1
  93. package/rust/core/audio/loader/trigger.rs +2 -1
  94. package/rust/core/audio/mod.rs +6 -7
  95. package/rust/core/audio/player.rs +70 -70
  96. package/rust/core/audio/special/easing.rs +189 -189
  97. package/rust/core/audio/special/env.rs +45 -45
  98. package/rust/core/audio/special/math.rs +134 -134
  99. package/rust/core/audio/special/mod.rs +9 -9
  100. package/rust/core/audio/special/modulator.rs +143 -143
  101. package/rust/core/builder/mod.rs +45 -2
  102. package/rust/core/debugger/lexer.rs +27 -27
  103. package/rust/core/debugger/{module.rs → logs.rs} +3 -6
  104. package/rust/core/debugger/mod.rs +30 -30
  105. package/rust/core/debugger/preprocessor.rs +27 -27
  106. package/rust/core/debugger/store.rs +2 -4
  107. package/rust/core/error/mod.rs +269 -269
  108. package/rust/core/lexer/driver.rs +59 -61
  109. package/rust/core/lexer/handler/arrow.rs +82 -82
  110. package/rust/core/lexer/handler/at.rs +21 -21
  111. package/rust/core/lexer/handler/brace.rs +41 -41
  112. package/rust/core/lexer/handler/colon.rs +21 -21
  113. package/rust/core/lexer/handler/comment.rs +30 -30
  114. package/rust/core/lexer/handler/dot.rs +21 -21
  115. package/rust/core/lexer/handler/driver.rs +337 -337
  116. package/rust/core/lexer/handler/identifier.rs +47 -47
  117. package/rust/core/lexer/handler/indent.rs +66 -66
  118. package/rust/core/lexer/handler/mod.rs +15 -15
  119. package/rust/core/lexer/handler/newline.rs +23 -23
  120. package/rust/core/lexer/handler/number.rs +31 -31
  121. package/rust/core/lexer/handler/operator.rs +46 -46
  122. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  123. package/rust/core/lexer/handler/slash.rs +21 -21
  124. package/rust/core/lexer/handler/string.rs +63 -63
  125. package/rust/core/lexer/mod.rs +3 -3
  126. package/rust/core/mod.rs +0 -1
  127. package/rust/core/parser/driver/block.rs +111 -0
  128. package/rust/core/parser/driver/cursor.rs +82 -0
  129. package/rust/core/parser/driver/driver_impl.rs +139 -0
  130. package/rust/core/parser/driver/mod.rs +6 -0
  131. package/rust/core/parser/driver/parse_array.rs +120 -0
  132. package/rust/core/parser/driver/parse_map.rs +223 -0
  133. package/rust/core/parser/driver/parser.rs +160 -0
  134. package/rust/core/parser/handler/arrow_call.rs +28 -4
  135. package/rust/core/parser/handler/at.rs +279 -279
  136. package/rust/core/parser/handler/bank.rs +104 -104
  137. package/rust/core/parser/handler/condition.rs +83 -83
  138. package/rust/core/parser/handler/dot.rs +148 -148
  139. package/rust/core/parser/handler/identifier/automate.rs +254 -254
  140. package/rust/core/parser/handler/identifier/call.rs +91 -91
  141. package/rust/core/parser/handler/identifier/emit.rs +70 -70
  142. package/rust/core/parser/handler/identifier/function.rs +113 -113
  143. package/rust/core/parser/handler/identifier/group.rs +89 -89
  144. package/rust/core/parser/handler/identifier/let_.rs +173 -173
  145. package/rust/core/parser/handler/identifier/mod.rs +55 -55
  146. package/rust/core/parser/handler/identifier/on.rs +107 -107
  147. package/rust/core/parser/handler/identifier/print.rs +49 -49
  148. package/rust/core/parser/handler/identifier/sleep.rs +96 -43
  149. package/rust/core/parser/handler/identifier/spawn.rs +91 -91
  150. package/rust/core/parser/handler/identifier/synth.rs +135 -135
  151. package/rust/core/parser/handler/loop_.rs +194 -194
  152. package/rust/core/parser/handler/mod.rs +9 -9
  153. package/rust/core/parser/handler/pattern.rs +1 -1
  154. package/rust/core/parser/handler/tempo.rs +105 -57
  155. package/rust/core/parser/statement.rs +10 -11
  156. package/rust/core/plugin/loader.rs +1 -1
  157. package/rust/core/plugin/mod.rs +2 -2
  158. package/rust/core/plugin/runner/mod.rs +11 -0
  159. package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +297 -347
  160. package/rust/core/plugin/runner/wasm32.rs +43 -0
  161. package/rust/core/preprocessor/loader/inject.rs +278 -0
  162. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  163. package/rust/core/preprocessor/loader/mod.rs +235 -0
  164. package/rust/core/preprocessor/module.rs +2 -7
  165. package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +6 -13
  166. package/rust/core/preprocessor/processor/mod.rs +1 -0
  167. package/rust/core/preprocessor/resolver/bank.rs +49 -49
  168. package/rust/core/preprocessor/resolver/call.rs +124 -124
  169. package/rust/core/preprocessor/resolver/condition.rs +95 -95
  170. package/rust/core/preprocessor/resolver/driver.rs +324 -324
  171. package/rust/core/preprocessor/resolver/function.rs +2 -2
  172. package/rust/core/preprocessor/resolver/group.rs +46 -18
  173. package/rust/core/preprocessor/resolver/let_.rs +32 -32
  174. package/rust/core/preprocessor/resolver/loop_.rs +318 -318
  175. package/rust/core/preprocessor/resolver/mod.rs +16 -16
  176. package/rust/core/preprocessor/resolver/pattern.rs +83 -83
  177. package/rust/core/preprocessor/resolver/spawn.rs +99 -99
  178. package/rust/core/preprocessor/resolver/synth.rs +54 -54
  179. package/rust/core/preprocessor/resolver/tempo.rs +48 -48
  180. package/rust/core/preprocessor/resolver/trigger.rs +116 -116
  181. package/rust/core/preprocessor/resolver/value.rs +176 -176
  182. package/rust/core/store/global.rs +2 -6
  183. package/rust/core/store/mod.rs +1 -5
  184. package/rust/lib.rs +18 -3
  185. package/rust/main.rs +27 -3
  186. package/rust/types/Cargo.toml +1 -1
  187. package/rust/types/src/addons.rs +55 -55
  188. package/rust/types/src/config.rs +84 -74
  189. package/rust/types/src/lib.rs +15 -12
  190. package/rust/types/src/plugin.rs +20 -0
  191. package/rust/types/src/store.rs +139 -0
  192. package/rust/types/src/telemetry.rs +85 -85
  193. package/rust/utils/Cargo.toml +2 -2
  194. package/rust/utils/src/file.rs +94 -94
  195. package/rust/utils/src/first_usage.rs +97 -97
  196. package/rust/utils/src/lib.rs +9 -9
  197. package/rust/utils/src/logger.rs +200 -200
  198. package/rust/utils/src/path.rs +129 -88
  199. package/rust/utils/src/signature.rs +41 -41
  200. package/rust/utils/src/spinner.rs +20 -20
  201. package/rust/utils/src/version.rs +27 -27
  202. package/rust/utils/src/watcher.rs +46 -46
  203. package/rust/web/api.rs +5 -5
  204. package/rust/web/cdn.rs +34 -34
  205. package/rust/web/mod.rs +3 -3
  206. package/tests/integration.rs +21 -21
  207. package/typescript/core/functions/index.ts +11 -0
  208. package/typescript/pkg/devalang_core.ts +20 -4
  209. package/rust/core/audio/engine/sample.rs +0 -366
  210. package/rust/core/audio/engine/synth.rs +0 -325
  211. package/rust/core/audio/interpreter/arrow_call.rs +0 -311
  212. package/rust/core/audio/renderer.rs +0 -54
  213. package/rust/core/parser/driver.rs +0 -584
  214. package/rust/core/preprocessor/loader.rs +0 -637
  215. package/rust/core/store/export.rs +0 -28
  216. package/rust/core/store/function.rs +0 -40
  217. package/rust/core/store/import.rs +0 -28
  218. package/rust/core/store/variable.rs +0 -51
  219. package/rust/core/utils/mod.rs +0 -1
  220. package/rust/core/utils/path.rs +0 -37
@@ -1,116 +1,116 @@
1
- use crate::core::{
2
- parser::statement::{Statement, StatementKind},
3
- preprocessor::module::Module,
4
- store::global::GlobalStore,
5
- };
6
- use devalang_types::{Duration, Value};
7
- use devalang_utils::logger::Logger;
8
- use std::collections::HashMap;
9
-
10
- pub fn resolve_trigger(
11
- stmt: &Statement,
12
- entity: &str,
13
- duration: &mut Duration,
14
- _effects: Option<Value>,
15
- module: &Module,
16
- path: &str,
17
- global_store: &GlobalStore,
18
- ) -> Statement {
19
- let logger = Logger::new();
20
-
21
- let mut final_duration = duration.clone();
22
-
23
- // Duration resolution
24
- if let Duration::Identifier(ident) = duration {
25
- if let Some(val) = resolve_identifier(ident, module, global_store) {
26
- match val {
27
- Value::Number(n) => {
28
- final_duration = Duration::Number(n);
29
- }
30
- Value::String(s) => {
31
- final_duration = Duration::Identifier(s);
32
- }
33
- Value::Identifier(s) if s == "auto" => {
34
- final_duration = Duration::Auto;
35
- }
36
- _ => {}
37
- }
38
- }
39
- }
40
-
41
- // Params value resolution
42
- let final_value = match &stmt.value {
43
- Value::Identifier(ident) => {
44
- logger.log_message(
45
- devalang_utils::logger::LogLevel::Debug,
46
- &format!("Resolving identifier: {}", ident),
47
- );
48
-
49
- match resolve_identifier(ident, module, global_store) {
50
- Some(v) => v,
51
- None => {
52
- logger.log_error_with_stacktrace(
53
- &format!("'{path}': value identifier '{ident}' not found"),
54
- &format!("{}:{}:{}", module.path, stmt.line, stmt.column),
55
- );
56
- Value::Null
57
- }
58
- }
59
- }
60
- Value::Map(map) => {
61
- let mut resolved_map = HashMap::new();
62
- for (k, v) in map {
63
- let resolved = match v {
64
- Value::Identifier(id) => {
65
- resolve_identifier(id, module, global_store).unwrap_or(Value::Null)
66
- }
67
- other => other.clone(),
68
- };
69
- resolved_map.insert(k.clone(), resolved);
70
- }
71
- Value::Map(resolved_map)
72
- }
73
- other => other.clone(),
74
- };
75
-
76
- Statement {
77
- kind: StatementKind::Trigger {
78
- entity: entity.to_string(),
79
- duration: final_duration.clone(),
80
- effects: Some(final_value.clone()),
81
- },
82
- value: Value::Null,
83
- line: stmt.line,
84
- column: stmt.column,
85
- indent: stmt.indent,
86
- }
87
- }
88
-
89
- fn resolve_identifier(ident: &str, module: &Module, global_store: &GlobalStore) -> Option<Value> {
90
- if let Some(val) = module.variable_table.get(ident) {
91
- return Some(resolve_value(val, module, global_store));
92
- }
93
-
94
- for other_mod in global_store.modules.values() {
95
- if let Some(val) = other_mod.variable_table.get(ident) {
96
- return Some(resolve_value(val, other_mod, global_store));
97
- }
98
- }
99
-
100
- None
101
- }
102
-
103
- fn resolve_value(val: &Value, module: &Module, global_store: &GlobalStore) -> Value {
104
- match val {
105
- Value::Identifier(inner) => resolve_identifier(inner, module, global_store)
106
- .unwrap_or(Value::Identifier(inner.clone())),
107
- Value::Map(map) => {
108
- let mut resolved = HashMap::new();
109
- for (k, v) in map {
110
- resolved.insert(k.clone(), resolve_value(v, module, global_store));
111
- }
112
- Value::Map(resolved)
113
- }
114
- other => other.clone(),
115
- }
116
- }
1
+ use crate::core::{
2
+ parser::statement::{Statement, StatementKind},
3
+ preprocessor::module::Module,
4
+ store::global::GlobalStore,
5
+ };
6
+ use devalang_types::{Duration, Value};
7
+ use devalang_utils::logger::Logger;
8
+ use std::collections::HashMap;
9
+
10
+ pub fn resolve_trigger(
11
+ stmt: &Statement,
12
+ entity: &str,
13
+ duration: &mut Duration,
14
+ _effects: Option<Value>,
15
+ module: &Module,
16
+ path: &str,
17
+ global_store: &GlobalStore,
18
+ ) -> Statement {
19
+ let logger = Logger::new();
20
+
21
+ let mut final_duration = duration.clone();
22
+
23
+ // Duration resolution
24
+ if let Duration::Identifier(ident) = duration {
25
+ if let Some(val) = resolve_identifier(ident, module, global_store) {
26
+ match val {
27
+ Value::Number(n) => {
28
+ final_duration = Duration::Number(n);
29
+ }
30
+ Value::String(s) => {
31
+ final_duration = Duration::Identifier(s);
32
+ }
33
+ Value::Identifier(s) if s == "auto" => {
34
+ final_duration = Duration::Auto;
35
+ }
36
+ _ => {}
37
+ }
38
+ }
39
+ }
40
+
41
+ // Params value resolution
42
+ let final_value = match &stmt.value {
43
+ Value::Identifier(ident) => {
44
+ logger.log_message(
45
+ devalang_utils::logger::LogLevel::Debug,
46
+ &format!("Resolving identifier: {}", ident),
47
+ );
48
+
49
+ match resolve_identifier(ident, module, global_store) {
50
+ Some(v) => v,
51
+ None => {
52
+ logger.log_error_with_stacktrace(
53
+ &format!("'{path}': value identifier '{ident}' not found"),
54
+ &format!("{}:{}:{}", module.path, stmt.line, stmt.column),
55
+ );
56
+ Value::Null
57
+ }
58
+ }
59
+ }
60
+ Value::Map(map) => {
61
+ let mut resolved_map = HashMap::new();
62
+ for (k, v) in map {
63
+ let resolved = match v {
64
+ Value::Identifier(id) => {
65
+ resolve_identifier(id, module, global_store).unwrap_or(Value::Null)
66
+ }
67
+ other => other.clone(),
68
+ };
69
+ resolved_map.insert(k.clone(), resolved);
70
+ }
71
+ Value::Map(resolved_map)
72
+ }
73
+ other => other.clone(),
74
+ };
75
+
76
+ Statement {
77
+ kind: StatementKind::Trigger {
78
+ entity: entity.to_string(),
79
+ duration: final_duration.clone(),
80
+ effects: Some(final_value.clone()),
81
+ },
82
+ value: Value::Null,
83
+ line: stmt.line,
84
+ column: stmt.column,
85
+ indent: stmt.indent,
86
+ }
87
+ }
88
+
89
+ fn resolve_identifier(ident: &str, module: &Module, global_store: &GlobalStore) -> Option<Value> {
90
+ if let Some(val) = module.variable_table.get(ident) {
91
+ return Some(resolve_value(val, module, global_store));
92
+ }
93
+
94
+ for other_mod in global_store.modules.values() {
95
+ if let Some(val) = other_mod.variable_table.get(ident) {
96
+ return Some(resolve_value(val, other_mod, global_store));
97
+ }
98
+ }
99
+
100
+ None
101
+ }
102
+
103
+ fn resolve_value(val: &Value, module: &Module, global_store: &GlobalStore) -> Value {
104
+ match val {
105
+ Value::Identifier(inner) => resolve_identifier(inner, module, global_store)
106
+ .unwrap_or(Value::Identifier(inner.clone())),
107
+ Value::Map(map) => {
108
+ let mut resolved = HashMap::new();
109
+ for (k, v) in map {
110
+ resolved.insert(k.clone(), resolve_value(v, module, global_store));
111
+ }
112
+ Value::Map(resolved)
113
+ }
114
+ other => other.clone(),
115
+ }
116
+ }
@@ -1,176 +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
+ 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,9 +1,5 @@
1
- use crate::core::{
2
- parser::statement::Statement,
3
- preprocessor::module::Module,
4
- store::{function::FunctionTable, variable::VariableTable},
5
- };
6
- use devalang_types::PluginInfo;
1
+ use crate::core::{parser::statement::Statement, preprocessor::module::Module};
2
+ use devalang_types::{FunctionTable, VariableTable, plugin::PluginInfo};
7
3
  use std::collections::HashMap;
8
4
 
9
5
  #[derive(Debug, Clone)]
@@ -1,5 +1 @@
1
- pub mod export;
2
- pub mod function;
3
- pub mod global;
4
- pub mod import;
5
- pub mod variable;
1
+ pub mod global;
package/rust/lib.rs CHANGED
@@ -5,10 +5,10 @@ use crate::core::{
5
5
  audio::{engine::AudioEngine, interpreter::driver::run_audio_program},
6
6
  parser::statement::{Statement, StatementKind},
7
7
  preprocessor::loader::ModuleLoader,
8
- store::{function::FunctionTable, global::GlobalStore, variable::VariableTable},
9
- utils::path::normalize_path,
8
+ store::global::GlobalStore,
10
9
  };
11
- use devalang_types::Value;
10
+ use devalang_types::{FunctionTable, Value, VariableTable};
11
+ use devalang_utils::path::normalize_path;
12
12
  use serde::{Deserialize, Serialize};
13
13
  use serde_wasm_bindgen::to_value;
14
14
  use wasm_bindgen::prelude::*;
@@ -171,6 +171,21 @@ pub fn register_playhead_callback(cb: &js_sys::Function) {
171
171
  }
172
172
  }
173
173
 
174
+ #[wasm_bindgen]
175
+ #[allow(unused_variables)]
176
+ pub fn collect_playhead_events() -> Result<JsValue, JsValue> {
177
+ #[cfg(target_arch = "wasm32")]
178
+ {
179
+ let events = crate::core::audio::interpreter::driver::collect_playhead_events();
180
+ to_value(&events).map_err(|e| JsValue::from_str(&format!("Error converting events: {}", e)))
181
+ }
182
+ #[cfg(not(target_arch = "wasm32"))]
183
+ {
184
+ // On non-wasm targets, return an empty array
185
+ to_value(&Vec::<String>::new()).map_err(|e| JsValue::from_str(&format!("Error: {}", e)))
186
+ }
187
+ }
188
+
174
189
  #[wasm_bindgen]
175
190
  pub fn unregister_playhead_callback() {
176
191
  #[cfg(target_arch = "wasm32")]