@devaloop/devalang 0.0.1-alpha.14 → 0.0.1-alpha.16

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 (177) hide show
  1. package/.devalang +10 -8
  2. package/.github/workflows/ci.yml +92 -0
  3. package/Cargo.toml +60 -58
  4. package/README.md +32 -15
  5. package/docs/CHANGELOG.md +93 -1
  6. package/docs/CONTRIBUTING.md +101 -1
  7. package/docs/ROADMAP.md +2 -2
  8. package/docs/TODO.md +1 -1
  9. package/examples/automation.deva +42 -0
  10. package/examples/bank.deva +4 -4
  11. package/examples/events.deva +12 -0
  12. package/examples/function.deva +4 -4
  13. package/examples/index.deva +39 -25
  14. package/examples/loop.deva +5 -11
  15. package/examples/pattern.deva +8 -0
  16. package/examples/plugin.deva +16 -0
  17. package/examples/variables.deva +1 -1
  18. package/out-tsc/bin/index.js +51 -7
  19. package/out-tsc/index.js +3 -1
  20. package/out-tsc/scripts/postbuild.js +9 -10
  21. package/out-tsc/scripts/postinstall.js +49 -0
  22. package/package.json +12 -4
  23. package/project-version.json +3 -3
  24. package/rust/cli/bank.rs +462 -456
  25. package/rust/cli/build.rs +252 -199
  26. package/rust/cli/check.rs +221 -180
  27. package/rust/cli/driver.rs +297 -292
  28. package/rust/cli/generator.rs +1 -0
  29. package/rust/cli/init.rs +87 -79
  30. package/rust/cli/install.rs +35 -32
  31. package/rust/cli/login.rs +127 -134
  32. package/rust/cli/mod.rs +13 -11
  33. package/rust/cli/play.rs +1123 -218
  34. package/rust/cli/telemetry.rs +19 -0
  35. package/rust/cli/template.rs +69 -57
  36. package/rust/cli/update.rs +6 -4
  37. package/rust/common/api.rs +5 -8
  38. package/rust/common/cdn.rs +3 -6
  39. package/rust/common/mod.rs +3 -3
  40. package/rust/common/sso.rs +3 -6
  41. package/rust/config/driver.rs +118 -94
  42. package/rust/config/loader.rs +165 -156
  43. package/rust/config/mod.rs +4 -2
  44. package/rust/config/settings.rs +91 -0
  45. package/rust/config/stats.rs +257 -0
  46. package/rust/core/audio/engine.rs +696 -518
  47. package/rust/core/audio/evaluator.rs +263 -31
  48. package/rust/core/audio/interpreter/arrow_call.rs +198 -161
  49. package/rust/core/audio/interpreter/automate.rs +18 -0
  50. package/rust/core/audio/interpreter/call.rs +98 -95
  51. package/rust/core/audio/interpreter/condition.rs +70 -71
  52. package/rust/core/audio/interpreter/driver.rs +487 -198
  53. package/rust/core/audio/interpreter/function.rs +26 -21
  54. package/rust/core/audio/interpreter/let_.rs +38 -19
  55. package/rust/core/audio/interpreter/load.rs +18 -18
  56. package/rust/core/audio/interpreter/loop_.rs +113 -73
  57. package/rust/core/audio/interpreter/mod.rs +14 -13
  58. package/rust/core/audio/interpreter/sleep.rs +27 -30
  59. package/rust/core/audio/interpreter/spawn.rs +105 -102
  60. package/rust/core/audio/interpreter/tempo.rs +19 -16
  61. package/rust/core/audio/interpreter/trigger.rs +239 -210
  62. package/rust/core/audio/loader/mod.rs +1 -1
  63. package/rust/core/audio/loader/trigger.rs +100 -97
  64. package/rust/core/audio/mod.rs +7 -6
  65. package/rust/core/audio/player.rs +64 -64
  66. package/rust/core/audio/renderer.rs +56 -53
  67. package/rust/core/audio/special/easing.rs +189 -0
  68. package/rust/core/audio/special/env.rs +43 -0
  69. package/rust/core/audio/special/math.rs +102 -0
  70. package/rust/core/audio/special/mod.rs +9 -0
  71. package/rust/core/audio/special/modulator.rs +143 -0
  72. package/rust/core/builder/mod.rs +80 -85
  73. package/rust/core/debugger/lexer.rs +27 -27
  74. package/rust/core/debugger/mod.rs +24 -23
  75. package/rust/core/debugger/module.rs +55 -47
  76. package/rust/core/debugger/preprocessor.rs +27 -27
  77. package/rust/core/debugger/store.rs +40 -39
  78. package/rust/core/error/mod.rs +80 -66
  79. package/rust/core/lexer/handler/arrow.rs +82 -31
  80. package/rust/core/lexer/handler/at.rs +21 -21
  81. package/rust/core/lexer/handler/brace.rs +41 -41
  82. package/rust/core/lexer/handler/colon.rs +21 -21
  83. package/rust/core/lexer/handler/comment.rs +30 -30
  84. package/rust/core/lexer/handler/dot.rs +21 -21
  85. package/rust/core/lexer/handler/driver.rs +337 -263
  86. package/rust/core/lexer/handler/identifier.rs +46 -42
  87. package/rust/core/lexer/handler/indent.rs +66 -66
  88. package/rust/core/lexer/handler/mod.rs +16 -16
  89. package/rust/core/lexer/handler/newline.rs +23 -23
  90. package/rust/core/lexer/handler/number.rs +31 -31
  91. package/rust/core/lexer/handler/operator.rs +46 -44
  92. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  93. package/rust/core/lexer/handler/slash.rs +21 -21
  94. package/rust/core/lexer/handler/string.rs +63 -63
  95. package/rust/core/lexer/mod.rs +54 -51
  96. package/rust/core/lexer/token.rs +97 -91
  97. package/rust/core/mod.rs +11 -11
  98. package/rust/core/parser/driver.rs +513 -408
  99. package/rust/core/parser/handler/arrow_call.rs +233 -211
  100. package/rust/core/parser/handler/at.rs +245 -162
  101. package/rust/core/parser/handler/bank.rs +94 -69
  102. package/rust/core/parser/handler/condition.rs +80 -74
  103. package/rust/core/parser/handler/dot.rs +143 -135
  104. package/rust/core/parser/handler/identifier/automate.rs +257 -0
  105. package/rust/core/parser/handler/identifier/call.rs +91 -88
  106. package/rust/core/parser/handler/identifier/emit.rs +66 -0
  107. package/rust/core/parser/handler/identifier/function.rs +100 -92
  108. package/rust/core/parser/handler/identifier/group.rs +85 -75
  109. package/rust/core/parser/handler/identifier/let_.rs +158 -127
  110. package/rust/core/parser/handler/identifier/mod.rs +54 -52
  111. package/rust/core/parser/handler/identifier/on.rs +98 -0
  112. package/rust/core/parser/handler/identifier/print.rs +52 -0
  113. package/rust/core/parser/handler/identifier/sleep.rs +36 -33
  114. package/rust/core/parser/handler/identifier/spawn.rs +91 -88
  115. package/rust/core/parser/handler/identifier/synth.rs +65 -65
  116. package/rust/core/parser/handler/loop_.rs +170 -72
  117. package/rust/core/parser/handler/mod.rs +8 -8
  118. package/rust/core/parser/handler/tempo.rs +53 -47
  119. package/rust/core/parser/mod.rs +4 -4
  120. package/rust/core/parser/statement.rs +142 -108
  121. package/rust/core/plugin/loader.rs +123 -48
  122. package/rust/core/plugin/mod.rs +2 -1
  123. package/rust/core/plugin/runner.rs +296 -0
  124. package/rust/core/preprocessor/loader.rs +515 -326
  125. package/rust/core/preprocessor/mod.rs +4 -4
  126. package/rust/core/preprocessor/module.rs +60 -58
  127. package/rust/core/preprocessor/processor.rs +99 -101
  128. package/rust/core/preprocessor/resolver/bank.rs +51 -49
  129. package/rust/core/preprocessor/resolver/call.rs +100 -100
  130. package/rust/core/preprocessor/resolver/condition.rs +97 -97
  131. package/rust/core/preprocessor/resolver/driver.rs +310 -278
  132. package/rust/core/preprocessor/resolver/function.rs +69 -78
  133. package/rust/core/preprocessor/resolver/group.rs +96 -91
  134. package/rust/core/preprocessor/resolver/let_.rs +32 -28
  135. package/rust/core/preprocessor/resolver/loop_.rs +320 -91
  136. package/rust/core/preprocessor/resolver/mod.rs +15 -15
  137. package/rust/core/preprocessor/resolver/spawn.rs +76 -92
  138. package/rust/core/preprocessor/resolver/synth.rs +56 -50
  139. package/rust/core/preprocessor/resolver/tempo.rs +50 -49
  140. package/rust/core/preprocessor/resolver/trigger.rs +113 -116
  141. package/rust/core/preprocessor/resolver/value.rs +81 -87
  142. package/rust/core/shared/bank.rs +1 -1
  143. package/rust/core/shared/duration.rs +9 -9
  144. package/rust/core/shared/mod.rs +3 -3
  145. package/rust/core/shared/value.rs +35 -32
  146. package/rust/core/store/function.rs +34 -34
  147. package/rust/core/store/global.rs +55 -38
  148. package/rust/core/store/mod.rs +5 -5
  149. package/rust/core/store/variable.rs +37 -34
  150. package/rust/core/utils/mod.rs +2 -2
  151. package/rust/core/utils/path.rs +37 -31
  152. package/rust/core/utils/validation.rs +35 -37
  153. package/rust/installer/addon.rs +84 -80
  154. package/rust/installer/bank.rs +62 -65
  155. package/rust/installer/mod.rs +5 -5
  156. package/rust/installer/plugin.rs +54 -55
  157. package/rust/installer/utils.rs +56 -56
  158. package/rust/lib.rs +156 -164
  159. package/rust/main.rs +250 -145
  160. package/rust/utils/error.rs +200 -0
  161. package/rust/utils/file.rs +38 -35
  162. package/rust/utils/first_usage.rs +76 -0
  163. package/rust/utils/logger.rs +195 -139
  164. package/rust/utils/mod.rs +9 -50
  165. package/rust/utils/signature.rs +19 -17
  166. package/rust/utils/spinner.rs +22 -19
  167. package/rust/utils/telemetry.rs +292 -0
  168. package/rust/utils/watcher.rs +34 -33
  169. package/templates/minimal/README.md +97 -121
  170. package/templates/welcome/README.md +97 -121
  171. package/typescript/bin/index.ts +19 -5
  172. package/typescript/index.ts +3 -1
  173. package/typescript/scripts/postbuild.ts +10 -6
  174. package/typescript/scripts/postinstall.ts +56 -0
  175. package/typescript/scripts/version/bump.ts +0 -1
  176. package/typescript/scripts/version/index.ts +0 -1
  177. package/out-tsc/bin/devalang.exe +0 -0
package/rust/main.rs CHANGED
@@ -1,145 +1,250 @@
1
- #![cfg(feature = "cli")]
2
-
3
- pub mod core;
4
- pub mod cli;
5
- pub mod utils;
6
- pub mod config;
7
- pub mod common;
8
- pub mod installer;
9
-
10
- use std::io;
11
- use clap::Parser;
12
- use crate::{
13
- cli::{
14
- bank::{
15
- handle_bank_available_command,
16
- handle_bank_info_command,
17
- handle_bank_list_command,
18
- handle_remove_bank_command,
19
- handle_update_bank_command,
20
- },
21
- build::handle_build_command,
22
- check::handle_check_command,
23
- driver::{ BankCommand, Cli, Commands, InstallCommand, TemplateCommand },
24
- init::handle_init_command,
25
- install::handle_install_command,
26
- login::handle_login_command,
27
- play::handle_play_command,
28
- template::{ handle_template_info_command, handle_template_list_command },
29
- update::handle_update_command,
30
- },
31
- config::{ driver::Config, loader::load_config },
32
- installer::addon::AddonType,
33
- };
34
-
35
- #[tokio::main]
36
- async fn main() -> io::Result<()> {
37
- let cli: Cli = Cli::parse();
38
- let mut config: Option<Config> = None;
39
-
40
- if !cli.no_config {
41
- config = load_config(None);
42
- } else {
43
- println!("No configuration file loaded. Running with arguments only.");
44
- }
45
-
46
- match cli.command {
47
- Commands::Init { name, template } => {
48
- handle_init_command(name, template);
49
- }
50
-
51
- Commands::Template { command } =>
52
- match command {
53
- TemplateCommand::List => {
54
- handle_template_list_command();
55
- }
56
- TemplateCommand::Info { name } => {
57
- handle_template_info_command(name);
58
- }
59
- }
60
-
61
- Commands::Check { entry, output, watch, debug } => {
62
- handle_check_command(config, entry, output, watch, debug);
63
- }
64
-
65
- Commands::Build { entry, output, watch, debug, compress } => {
66
- handle_build_command(config, entry, output, watch, debug, compress);
67
- }
68
-
69
- Commands::Play { entry, output, watch, repeat, debug } => {
70
- handle_play_command(config, entry, output, watch, repeat, debug);
71
- }
72
-
73
- Commands::Install { command } =>
74
- match command {
75
- InstallCommand::Bank { name } => {
76
- if let Err(err) = handle_install_command(name, AddonType::Bank).await {
77
- eprintln!("❌ Failed to install bank: {}", err);
78
- }
79
- }
80
- InstallCommand::Plugin { name } => {
81
- if let Err(err) = handle_install_command(name, AddonType::Plugin).await {
82
- eprintln!("❌ Failed to install plugin: {}", err);
83
- }
84
- }
85
- InstallCommand::Preset { name } => {
86
- if let Err(err) = handle_install_command(name, AddonType::Preset).await {
87
- eprintln!("❌ Failed to install preset: {}", err);
88
- }
89
- }
90
- }
91
-
92
- Commands::Bank { command } =>
93
- match command {
94
- BankCommand::List => {
95
- if let Err(err) = handle_bank_list_command().await {
96
- eprintln!("❌ Failed to list local banks: {}", err);
97
- }
98
- }
99
-
100
- BankCommand::Available => {
101
- if let Err(err) = handle_bank_available_command().await {
102
- eprintln!("❌ Failed to list available banks: {}", err);
103
- }
104
- }
105
-
106
- BankCommand::Info { name } => {
107
- if let Err(err) = handle_bank_info_command(name).await {
108
- eprintln!(" Failed to get bank info: {}", err);
109
- }
110
- }
111
-
112
- BankCommand::Remove { name } => {
113
- if let Err(err) = handle_remove_bank_command(name).await {
114
- eprintln!("❌ Failed to remove bank: {}", err);
115
- }
116
- }
117
-
118
- BankCommand::Update { name } => {
119
- if let Err(err) = handle_update_bank_command(name).await {
120
- eprintln!("❌ Failed to update bank: {}", err);
121
- }
122
- }
123
- }
124
-
125
- Commands::Update { only } => {
126
- if let Err(err) = handle_update_command(only).await {
127
- eprintln!("❌ Update failed: {}", err);
128
- }
129
- }
130
-
131
- Commands::Login { .. } => {
132
- if let Err(err) = handle_login_command().await {
133
- eprintln!("❌ Login failed: {}", err);
134
- }
135
- }
136
-
137
- Commands::Logout { .. } => {
138
- eprintln!("❌ Logout command is not implemented yet.");
139
- }
140
-
141
- _ => {}
142
- }
143
-
144
- Ok(())
145
- }
1
+ #![cfg(feature = "cli")]
2
+
3
+ pub mod cli;
4
+ pub mod common;
5
+ pub mod config;
6
+ pub mod core;
7
+ pub mod installer;
8
+ pub mod utils;
9
+
10
+ use crate::{
11
+ cli::{
12
+ bank::{
13
+ handle_bank_available_command, handle_bank_info_command, handle_bank_list_command,
14
+ handle_remove_bank_command, handle_update_bank_command,
15
+ },
16
+ build::handle_build_command,
17
+ check::handle_check_command,
18
+ driver::{BankCommand, Cli, Commands, InstallCommand, TelemetryCommand, TemplateCommand},
19
+ init::handle_init_command,
20
+ install::handle_install_command,
21
+ login::handle_login_command,
22
+ play::handle_play_command,
23
+ telemetry::{handle_telemetry_disable_command, handle_telemetry_enable_command},
24
+ template::{handle_template_info_command, handle_template_list_command},
25
+ update::handle_update_command,
26
+ },
27
+ config::{driver::ProjectConfig, loader::load_config},
28
+ installer::addon::AddonType,
29
+ utils::{first_usage::check_is_first_usage, telemetry::TelemetryEventCreator},
30
+ };
31
+ use clap::Parser;
32
+ use std::io;
33
+
34
+ #[tokio::main]
35
+ async fn main() -> io::Result<()> {
36
+ let cli: Cli = Cli::parse();
37
+ let mut config: Option<ProjectConfig> = None;
38
+
39
+ let duration = std::time::Instant::now();
40
+
41
+ check_is_first_usage();
42
+
43
+ let telemetry_event_creator = TelemetryEventCreator::new();
44
+ let mut event = telemetry_event_creator.get_base_event();
45
+ let mut had_error: bool = false;
46
+ let mut last_error_message: Option<String> = None;
47
+ let mut exit_code: Option<i32> = None;
48
+
49
+ if !cli.no_config {
50
+ config = load_config(None);
51
+ } else {
52
+ println!("No configuration file loaded. Running with arguments only.");
53
+ }
54
+
55
+ match cli.command {
56
+ Commands::Init { name, template } => {
57
+ handle_init_command(name, template);
58
+ }
59
+
60
+ Commands::Template { command } => match command {
61
+ TemplateCommand::List => {
62
+ handle_template_list_command();
63
+ }
64
+ TemplateCommand::Info { name } => {
65
+ handle_template_info_command(name);
66
+ }
67
+ },
68
+
69
+ Commands::Check {
70
+ entry,
71
+ output,
72
+ watch,
73
+ debug,
74
+ } => {
75
+ if let Err(err) = handle_check_command(config, entry, output, watch, debug) {
76
+ eprintln!("❌ Check failed: {}", err);
77
+ had_error = true;
78
+ last_error_message = Some(format!("check failed: {}", err));
79
+ exit_code = Some(1);
80
+ }
81
+ }
82
+
83
+ Commands::Build {
84
+ entry,
85
+ output,
86
+ watch,
87
+ debug,
88
+ compress,
89
+ } => {
90
+ if let Err(err) = handle_build_command(config, entry, output, watch, debug, compress) {
91
+ eprintln!("❌ Build failed: {}", err);
92
+ had_error = true;
93
+ last_error_message = Some(format!("build failed: {}", err));
94
+ exit_code = Some(1);
95
+ }
96
+ }
97
+
98
+ Commands::Play {
99
+ entry,
100
+ output,
101
+ watch,
102
+ repeat,
103
+ debug,
104
+ } => {
105
+ if let Err(err) = handle_play_command(config, entry, output, watch, repeat, debug) {
106
+ eprintln!("❌ Play failed: {}", err);
107
+ had_error = true;
108
+ last_error_message = Some(format!("play failed: {}", err));
109
+ exit_code = Some(1);
110
+ }
111
+ }
112
+
113
+ Commands::Install { command } => match command {
114
+ InstallCommand::Bank { name } => {
115
+ if let Err(err) = handle_install_command(name, AddonType::Bank).await {
116
+ eprintln!("❌ Failed to install bank: {}", err);
117
+ had_error = true;
118
+ last_error_message = Some(format!("install bank failed: {}", err));
119
+ exit_code = Some(1);
120
+ }
121
+ }
122
+ InstallCommand::Plugin { name } => {
123
+ if let Err(err) = handle_install_command(name, AddonType::Plugin).await {
124
+ eprintln!("❌ Failed to install plugin: {}", err);
125
+ had_error = true;
126
+ last_error_message = Some(format!("install plugin failed: {}", err));
127
+ exit_code = Some(1);
128
+ }
129
+ }
130
+ InstallCommand::Preset { name } => {
131
+ if let Err(err) = handle_install_command(name, AddonType::Preset).await {
132
+ eprintln!("❌ Failed to install preset: {}", err);
133
+ had_error = true;
134
+ last_error_message = Some(format!("install preset failed: {}", err));
135
+ exit_code = Some(1);
136
+ }
137
+ }
138
+ },
139
+
140
+ Commands::Bank { command } => match command {
141
+ BankCommand::List => {
142
+ if let Err(err) = handle_bank_list_command().await {
143
+ eprintln!("❌ Failed to list local banks: {}", err);
144
+ had_error = true;
145
+ last_error_message = Some(format!("bank list failed: {}", err));
146
+ exit_code = Some(1);
147
+ }
148
+ }
149
+
150
+ BankCommand::Available => {
151
+ if let Err(err) = handle_bank_available_command().await {
152
+ eprintln!("❌ Failed to list available banks: {}", err);
153
+ had_error = true;
154
+ last_error_message = Some(format!("bank available failed: {}", err));
155
+ exit_code = Some(1);
156
+ }
157
+ }
158
+
159
+ BankCommand::Info { name } => {
160
+ if let Err(err) = handle_bank_info_command(name).await {
161
+ eprintln!("❌ Failed to get bank info: {}", err);
162
+ had_error = true;
163
+ last_error_message = Some(format!("bank info failed: {}", err));
164
+ exit_code = Some(1);
165
+ }
166
+ }
167
+
168
+ BankCommand::Remove { name } => {
169
+ if let Err(err) = handle_remove_bank_command(name).await {
170
+ eprintln!("❌ Failed to remove bank: {}", err);
171
+ had_error = true;
172
+ last_error_message = Some(format!("bank remove failed: {}", err));
173
+ exit_code = Some(1);
174
+ }
175
+ }
176
+
177
+ BankCommand::Update { name } => {
178
+ if let Err(err) = handle_update_bank_command(name).await {
179
+ eprintln!("❌ Failed to update bank: {}", err);
180
+ had_error = true;
181
+ last_error_message = Some(format!("bank update failed: {}", err));
182
+ exit_code = Some(1);
183
+ }
184
+ }
185
+ },
186
+
187
+ Commands::Update { only } => {
188
+ if let Err(err) = handle_update_command(only).await {
189
+ eprintln!("❌ Update failed: {}", err);
190
+ had_error = true;
191
+ last_error_message = Some(format!("update failed: {}", err));
192
+ exit_code = Some(1);
193
+ }
194
+ }
195
+
196
+ Commands::Telemetry { command } => match command {
197
+ TelemetryCommand::Enable { .. } => {
198
+ if let Err(err) = handle_telemetry_enable_command().await {
199
+ eprintln!("❌ Failed to enable telemetry: {}", err);
200
+ had_error = true;
201
+ last_error_message = Some(format!("telemetry enable failed: {}", err));
202
+ exit_code = Some(1);
203
+ }
204
+ }
205
+ TelemetryCommand::Disable { .. } => {
206
+ if let Err(err) = handle_telemetry_disable_command().await {
207
+ eprintln!("❌ Failed to disable telemetry: {}", err);
208
+ had_error = true;
209
+ last_error_message = Some(format!("telemetry disable failed: {}", err));
210
+ exit_code = Some(1);
211
+ }
212
+ }
213
+ },
214
+
215
+ Commands::Login { .. } => {
216
+ if let Err(err) = handle_login_command().await {
217
+ eprintln!("❌ Login failed: {}", err);
218
+ had_error = true;
219
+ last_error_message = Some(format!("login failed: {}", err));
220
+ exit_code = Some(1);
221
+ }
222
+ }
223
+
224
+ Commands::Logout { .. } => {
225
+ eprintln!("❌ Logout command is not implemented yet.");
226
+ had_error = true;
227
+ last_error_message = Some("logout not implemented".to_string());
228
+ exit_code = Some(1);
229
+ }
230
+ }
231
+
232
+ // SECTION Telemetry
233
+ event.set_timestamp(chrono::Utc::now().to_string());
234
+ event.set_duration(duration.elapsed().as_millis() as u64);
235
+ event.set_success(!had_error);
236
+
237
+ if had_error {
238
+ event.set_error(
239
+ utils::telemetry::TelemetryErrorLevel::Critical,
240
+ last_error_message,
241
+ exit_code,
242
+ );
243
+ }
244
+
245
+ utils::telemetry::refresh_event_project_info(&mut event);
246
+
247
+ let _ = utils::telemetry::send_telemetry_event(&event).await;
248
+
249
+ Ok(())
250
+ }
@@ -0,0 +1,200 @@
1
+ use super::logger::{LogLevel, Logger};
2
+ use crate::core::{
3
+ error::{ErrorResult, Severity, StackFrame},
4
+ parser::statement::{Statement, StatementKind},
5
+ shared::value::Value,
6
+ };
7
+ use std::collections::HashMap;
8
+
9
+ /// Recursively collects errors from a list of statements.
10
+ ///
11
+ /// This function traverses the provided statements and aggregates any
12
+ /// `Unknown` or explicit `Error` statements into a flat vector.
13
+ /// It also descends into loop bodies to ensure nested errors are
14
+ /// surfaced.
15
+ pub fn collect_errors_recursively(statements: &[Statement]) -> Vec<ErrorResult> {
16
+ let mut errors: Vec<ErrorResult> = Vec::new();
17
+
18
+ for stmt in statements {
19
+ match &stmt.kind {
20
+ StatementKind::Unknown => {
21
+ errors.push(ErrorResult {
22
+ message: format!("Unknown statement at line {}:{}", stmt.line, stmt.column),
23
+ line: stmt.line,
24
+ column: stmt.column,
25
+ severity: Severity::Warning,
26
+ stack: vec![StackFrame {
27
+ module: None,
28
+ context: Some("Unknown".to_string()),
29
+ line: stmt.line,
30
+ column: stmt.column,
31
+ }],
32
+ });
33
+ }
34
+ StatementKind::Error { message } => {
35
+ errors.push(ErrorResult {
36
+ message: message.clone(),
37
+ line: stmt.line,
38
+ column: stmt.column,
39
+ severity: Severity::Critical,
40
+ stack: vec![StackFrame {
41
+ module: None,
42
+ context: Some("Error".to_string()),
43
+ line: stmt.line,
44
+ column: stmt.column,
45
+ }],
46
+ });
47
+ }
48
+ StatementKind::Loop => {
49
+ if let Some(body_statements) = extract_loop_body_statements(&stmt.value) {
50
+ let nested = collect_errors_recursively(body_statements);
51
+ errors.extend(nested.into_iter().map(|mut e| {
52
+ // push parent frame contextually for a basic callstack-like trace
53
+ e.stack.insert(
54
+ 0,
55
+ StackFrame {
56
+ module: None,
57
+ context: Some("loop".to_string()),
58
+ line: stmt.line,
59
+ column: stmt.column,
60
+ },
61
+ );
62
+ e
63
+ }));
64
+ }
65
+ }
66
+ _ => {}
67
+ }
68
+ }
69
+
70
+ errors
71
+ }
72
+
73
+ fn extract_loop_body_statements(value: &Value) -> Option<&[Statement]> {
74
+ if let Value::Map(map) = value {
75
+ if let Some(Value::Block(statements)) = map.get("body") {
76
+ return Some(statements);
77
+ }
78
+ }
79
+ None
80
+ }
81
+
82
+ pub fn partition_errors(errors: Vec<ErrorResult>) -> (Vec<ErrorResult>, Vec<ErrorResult>) {
83
+ let mut warnings = Vec::new();
84
+ let mut criticals = Vec::new();
85
+ for e in errors {
86
+ match e.severity {
87
+ Severity::Warning => warnings.push(e),
88
+ Severity::Critical => criticals.push(e),
89
+ }
90
+ }
91
+ (warnings, criticals)
92
+ }
93
+
94
+ pub fn log_errors_with_stack(prefix: &str, warnings: &[ErrorResult], criticals: &[ErrorResult]) {
95
+ let logger = Logger::new();
96
+ if !warnings.is_empty() {
97
+ logger.log_message(
98
+ LogLevel::Warning,
99
+ &format!("{}: {} warning(s)", prefix, warnings.len()),
100
+ );
101
+ for w in warnings {
102
+ logger.log_message(LogLevel::Warning, &format!("- {}", w.message));
103
+ // Print primary location in path:line:col if available
104
+ if let Some(frame) = w.stack.first() {
105
+ let module = frame.module.clone().unwrap_or_default();
106
+ logger.log_message(
107
+ LogLevel::Debug,
108
+ &format!(
109
+ " ↳ {}:{}:{} {}",
110
+ module,
111
+ frame.line,
112
+ frame.column,
113
+ frame.context.clone().unwrap_or_default()
114
+ ),
115
+ );
116
+ }
117
+ // Print remaining frames if any
118
+ if w.stack.len() > 1 {
119
+ for (i, f) in w.stack.iter().enumerate().skip(1) {
120
+ let module = f.module.clone().unwrap_or_default();
121
+ logger.log_message(
122
+ LogLevel::Debug,
123
+ &format!(
124
+ " #{} {}:{}:{} {}",
125
+ i,
126
+ module,
127
+ f.line,
128
+ f.column,
129
+ f.context.clone().unwrap_or_default()
130
+ ),
131
+ );
132
+ }
133
+ }
134
+ }
135
+ }
136
+ if !criticals.is_empty() {
137
+ logger.log_message(
138
+ LogLevel::Error,
139
+ &format!("{}: {} critical error(s)", prefix, criticals.len()),
140
+ );
141
+ for c in criticals {
142
+ logger.log_message(LogLevel::Error, &format!("- {}", c.message));
143
+ if let Some(frame) = c.stack.first() {
144
+ let module = frame.module.clone().unwrap_or_default();
145
+ logger.log_message(
146
+ LogLevel::Error,
147
+ &format!(
148
+ " ↳ {}:{}:{} {}",
149
+ module,
150
+ frame.line,
151
+ frame.column,
152
+ frame.context.clone().unwrap_or_default()
153
+ ),
154
+ );
155
+ }
156
+ if c.stack.len() > 1 {
157
+ for (i, f) in c.stack.iter().enumerate().skip(1) {
158
+ let module = f.module.clone().unwrap_or_default();
159
+ logger.log_message(
160
+ LogLevel::Error,
161
+ &format!(
162
+ " #{} {}:{}:{} {}",
163
+ i,
164
+ module,
165
+ f.line,
166
+ f.column,
167
+ f.context.clone().unwrap_or_default()
168
+ ),
169
+ );
170
+ }
171
+ }
172
+ }
173
+ }
174
+ }
175
+
176
+ pub fn collect_all_errors_with_modules(
177
+ modules: &HashMap<String, Vec<Statement>>,
178
+ ) -> Vec<ErrorResult> {
179
+ let mut all = Vec::new();
180
+ for (module_path, stmts) in modules {
181
+ let mut errs = collect_errors_recursively(stmts);
182
+ for e in errs.iter_mut() {
183
+ // ensure top frame carries the module path
184
+ if e.stack.is_empty() {
185
+ e.stack.push(StackFrame {
186
+ module: Some(module_path.clone()),
187
+ context: None,
188
+ line: e.line,
189
+ column: e.column,
190
+ });
191
+ } else {
192
+ if e.stack[0].module.is_none() {
193
+ e.stack[0].module = Some(module_path.clone());
194
+ }
195
+ }
196
+ }
197
+ all.extend(errs);
198
+ }
199
+ all
200
+ }
@@ -1,35 +1,38 @@
1
- use std::{ fs::{ self }, path::Path };
2
- use include_dir::{ Dir, DirEntry };
3
-
4
- pub fn copy_dir_recursive(dir: &Dir, target_root: &Path, base_path: &Path) {
5
- for entry in dir.entries() {
6
- match entry {
7
- DirEntry::Dir(subdir) => {
8
- copy_dir_recursive(subdir, target_root, base_path);
9
- }
10
- DirEntry::File(file) => {
11
- let rel_path = file.path().strip_prefix(base_path).unwrap();
12
- let dest_path = target_root.join(rel_path);
13
-
14
- if let Some(parent) = dest_path.parent() {
15
- fs::create_dir_all(parent).unwrap();
16
- }
17
-
18
- fs::write(&dest_path, file.contents()).expect("Error writing file");
19
- }
20
- }
21
- }
22
- }
23
-
24
- pub fn format_file_size(bytes: u64) -> String {
25
- const KB: u64 = 1024;
26
- const MB: u64 = 1024 * 1024;
27
-
28
- if bytes >= MB {
29
- format!("{:.2} Mb", (bytes as f64) / (MB as f64))
30
- } else if bytes >= KB {
31
- format!("{:.2} Kb", (bytes as f64) / (KB as f64))
32
- } else {
33
- format!("{} bytes", bytes)
34
- }
35
- }
1
+ use include_dir::{Dir, DirEntry};
2
+ use std::{
3
+ fs::{self},
4
+ path::Path,
5
+ };
6
+
7
+ pub fn copy_dir_recursive(dir: &Dir, target_root: &Path, base_path: &Path) {
8
+ for entry in dir.entries() {
9
+ match entry {
10
+ DirEntry::Dir(subdir) => {
11
+ copy_dir_recursive(subdir, target_root, base_path);
12
+ }
13
+ DirEntry::File(file) => {
14
+ let rel_path = file.path().strip_prefix(base_path).unwrap();
15
+ let dest_path = target_root.join(rel_path);
16
+
17
+ if let Some(parent) = dest_path.parent() {
18
+ fs::create_dir_all(parent).unwrap();
19
+ }
20
+
21
+ fs::write(&dest_path, file.contents()).expect("Error writing file");
22
+ }
23
+ }
24
+ }
25
+ }
26
+
27
+ pub fn format_file_size(bytes: u64) -> String {
28
+ const KB: u64 = 1024;
29
+ const MB: u64 = 1024 * 1024;
30
+
31
+ if bytes >= MB {
32
+ format!("{:.2} Mb", (bytes as f64) / (MB as f64))
33
+ } else if bytes >= KB {
34
+ format!("{:.2} Kb", (bytes as f64) / (KB as f64))
35
+ } else {
36
+ format!("{} bytes", bytes)
37
+ }
38
+ }