@devaloop/devalang 0.0.1-alpha.16-hotfix.1 → 0.0.1-alpha.17

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 (235) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +6 -10
  3. package/.github/workflows/ci.yml +19 -8
  4. package/Cargo.toml +18 -2
  5. package/README.md +80 -33
  6. package/docs/CHANGELOG.md +56 -0
  7. package/docs/ROADMAP.md +6 -3
  8. package/examples/index.deva +52 -35
  9. package/out-tsc/bin/index.d.ts +2 -0
  10. package/out-tsc/core/functions/index.d.ts +37 -0
  11. package/out-tsc/core/functions/index.js +76 -0
  12. package/out-tsc/core/index.d.ts +6 -0
  13. package/out-tsc/core/index.js +22 -0
  14. package/out-tsc/core/types/index.d.ts +4 -0
  15. package/out-tsc/core/types/index.js +20 -0
  16. package/out-tsc/core/types/plugin.d.ts +18 -0
  17. package/out-tsc/core/types/plugin.js +2 -0
  18. package/out-tsc/core/types/result.d.ts +27 -0
  19. package/out-tsc/core/types/result.js +2 -0
  20. package/out-tsc/core/types/statement.d.ts +106 -0
  21. package/out-tsc/core/types/statement.js +2 -0
  22. package/out-tsc/core/types/value.d.ts +43 -0
  23. package/out-tsc/core/types/value.js +2 -0
  24. package/out-tsc/index.d.ts +7 -0
  25. package/out-tsc/index.js +41 -2
  26. package/out-tsc/pkg/devalang_core.d.ts +7 -0
  27. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +33 -0
  28. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  29. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  30. package/out-tsc/scripts/postinstall.d.ts +1 -0
  31. package/out-tsc/scripts/postinstall.js +33 -23
  32. package/out-tsc/scripts/version/bump.d.ts +1 -0
  33. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  34. package/out-tsc/scripts/version/index.d.ts +1 -0
  35. package/out-tsc/scripts/version/sync.d.ts +1 -0
  36. package/package.json +16 -4
  37. package/project-version.json +3 -3
  38. package/rust/cli/bank/api.rs +122 -0
  39. package/rust/cli/bank/commands.rs +275 -0
  40. package/rust/cli/bank/mod.rs +29 -0
  41. package/rust/cli/build/commands.rs +97 -0
  42. package/rust/cli/build/mod.rs +2 -0
  43. package/rust/cli/build/process.rs +146 -0
  44. package/rust/cli/{check.rs → check/mod.rs} +18 -31
  45. package/rust/cli/discover/commands.rs +253 -0
  46. package/rust/cli/discover/config.rs +111 -0
  47. package/rust/cli/discover/fs.rs +19 -0
  48. package/rust/cli/discover/install.rs +103 -0
  49. package/rust/cli/discover/metadata.rs +48 -0
  50. package/rust/cli/discover/mod.rs +5 -0
  51. package/rust/cli/{init.rs → init/commands.rs} +88 -87
  52. package/rust/cli/init/mod.rs +1 -0
  53. package/rust/{installer → cli/install}/addon.rs +5 -9
  54. package/rust/cli/install/bank.rs +53 -0
  55. package/rust/cli/{install.rs → install/commands.rs} +9 -9
  56. package/rust/{installer → cli/install}/mod.rs +2 -3
  57. package/rust/cli/install/plugin.rs +61 -0
  58. package/rust/cli/{login.rs → login/commands.rs} +8 -11
  59. package/rust/cli/login/mod.rs +1 -0
  60. package/rust/cli/mod.rs +2 -3
  61. package/rust/cli/{driver.rs → parser.rs} +19 -2
  62. package/rust/cli/play/commands.rs +324 -0
  63. package/rust/cli/play/io.rs +17 -0
  64. package/rust/cli/play/mod.rs +5 -0
  65. package/rust/cli/play/process.rs +150 -0
  66. package/rust/cli/play/realtime.rs +91 -0
  67. package/rust/cli/play/utils.rs +23 -0
  68. package/rust/cli/telemetry/commands.rs +22 -0
  69. package/rust/cli/telemetry/event_creator.rs +80 -0
  70. package/rust/cli/telemetry/mod.rs +3 -0
  71. package/rust/cli/telemetry/send.rs +51 -0
  72. package/rust/cli/{template.rs → template/commands.rs} +1 -1
  73. package/rust/cli/template/mod.rs +1 -0
  74. package/rust/cli/{update.rs → update/commands.rs} +6 -6
  75. package/rust/cli/update/mod.rs +1 -0
  76. package/rust/config/driver.rs +57 -72
  77. package/rust/config/mod.rs +1 -2
  78. package/rust/config/ops.rs +26 -0
  79. package/rust/config/settings.rs +60 -50
  80. package/rust/core/audio/engine/helpers.rs +146 -0
  81. package/rust/core/audio/engine/mod.rs +7 -0
  82. package/rust/core/audio/engine/sample.rs +298 -0
  83. package/rust/core/audio/engine/synth.rs +310 -0
  84. package/rust/core/audio/evaluator.rs +15 -12
  85. package/rust/core/audio/interpreter/arrow_call.rs +99 -24
  86. package/rust/core/audio/interpreter/call.rs +81 -60
  87. package/rust/core/audio/interpreter/condition.rs +3 -2
  88. package/rust/core/audio/interpreter/driver.rs +206 -151
  89. package/rust/core/audio/interpreter/let_.rs +1 -1
  90. package/rust/core/audio/interpreter/load.rs +2 -1
  91. package/rust/core/audio/interpreter/loop_.rs +7 -6
  92. package/rust/core/audio/interpreter/sleep.rs +2 -1
  93. package/rust/core/audio/interpreter/spawn.rs +45 -57
  94. package/rust/core/audio/interpreter/tempo.rs +31 -10
  95. package/rust/core/audio/interpreter/trigger.rs +2 -2
  96. package/rust/core/audio/loader/trigger.rs +4 -7
  97. package/rust/core/audio/player.rs +6 -0
  98. package/rust/core/audio/renderer.rs +5 -7
  99. package/rust/core/audio/special/env.rs +3 -1
  100. package/rust/core/audio/special/math.rs +4 -4
  101. package/rust/core/audio/special/modulator.rs +2 -2
  102. package/rust/core/builder/mod.rs +9 -3
  103. package/rust/core/debugger/lexer.rs +1 -1
  104. package/rust/core/debugger/mod.rs +6 -0
  105. package/rust/core/debugger/module.rs +4 -4
  106. package/rust/core/debugger/preprocessor.rs +1 -1
  107. package/rust/core/debugger/store.rs +2 -2
  108. package/rust/core/error/mod.rs +189 -0
  109. package/rust/core/lexer/handler/arrow.rs +1 -1
  110. package/rust/core/lexer/handler/at.rs +1 -1
  111. package/rust/core/lexer/handler/brace.rs +2 -2
  112. package/rust/core/lexer/handler/colon.rs +1 -1
  113. package/rust/core/lexer/handler/comment.rs +1 -1
  114. package/rust/core/lexer/handler/dot.rs +1 -1
  115. package/rust/core/lexer/handler/driver.rs +1 -1
  116. package/rust/core/lexer/handler/identifier.rs +1 -1
  117. package/rust/core/lexer/handler/mod.rs +1 -2
  118. package/rust/core/lexer/handler/number.rs +1 -1
  119. package/rust/core/lexer/handler/operator.rs +1 -1
  120. package/rust/core/lexer/handler/parenthesis.rs +2 -2
  121. package/rust/core/lexer/handler/slash.rs +1 -1
  122. package/rust/core/lexer/handler/string.rs +1 -1
  123. package/rust/core/lexer/mod.rs +22 -12
  124. package/rust/core/lexer/token.rs +90 -97
  125. package/rust/core/mod.rs +0 -1
  126. package/rust/core/parser/driver.rs +66 -13
  127. package/rust/core/parser/handler/arrow_call.rs +28 -8
  128. package/rust/core/parser/handler/at.rs +55 -21
  129. package/rust/core/parser/handler/bank.rs +14 -4
  130. package/rust/core/parser/handler/condition.rs +6 -3
  131. package/rust/core/parser/handler/dot.rs +2 -1
  132. package/rust/core/parser/handler/identifier/automate.rs +13 -16
  133. package/rust/core/parser/handler/identifier/call.rs +4 -4
  134. package/rust/core/parser/handler/identifier/emit.rs +9 -5
  135. package/rust/core/parser/handler/identifier/function.rs +20 -7
  136. package/rust/core/parser/handler/identifier/group.rs +11 -7
  137. package/rust/core/parser/handler/identifier/let_.rs +24 -9
  138. package/rust/core/parser/handler/identifier/mod.rs +6 -5
  139. package/rust/core/parser/handler/identifier/on.rs +16 -7
  140. package/rust/core/parser/handler/identifier/print.rs +6 -9
  141. package/rust/core/parser/handler/identifier/sleep.rs +12 -5
  142. package/rust/core/parser/handler/identifier/spawn.rs +4 -4
  143. package/rust/core/parser/handler/identifier/synth.rs +79 -9
  144. package/rust/core/parser/handler/loop_.rs +39 -14
  145. package/rust/core/parser/handler/tempo.rs +9 -5
  146. package/rust/core/parser/mod.rs +0 -1
  147. package/rust/core/parser/statement.rs +6 -137
  148. package/rust/core/plugin/loader.rs +41 -27
  149. package/rust/core/plugin/runner.rs +68 -17
  150. package/rust/core/preprocessor/loader.rs +155 -33
  151. package/rust/core/preprocessor/processor.rs +2 -2
  152. package/rust/core/preprocessor/resolver/bank.rs +6 -8
  153. package/rust/core/preprocessor/resolver/call.rs +20 -24
  154. package/rust/core/preprocessor/resolver/condition.rs +6 -8
  155. package/rust/core/preprocessor/resolver/driver.rs +14 -16
  156. package/rust/core/preprocessor/resolver/function.rs +6 -6
  157. package/rust/core/preprocessor/resolver/group.rs +6 -8
  158. package/rust/core/preprocessor/resolver/loop_.rs +8 -10
  159. package/rust/core/preprocessor/resolver/spawn.rs +19 -23
  160. package/rust/core/preprocessor/resolver/synth.rs +6 -8
  161. package/rust/core/preprocessor/resolver/tempo.rs +6 -8
  162. package/rust/core/preprocessor/resolver/trigger.rs +22 -19
  163. package/rust/core/preprocessor/resolver/value.rs +99 -4
  164. package/rust/core/store/export.rs +28 -28
  165. package/rust/core/store/function.rs +6 -0
  166. package/rust/core/store/global.rs +7 -1
  167. package/rust/core/store/import.rs +28 -28
  168. package/rust/core/store/variable.rs +1 -1
  169. package/rust/core/utils/mod.rs +0 -1
  170. package/rust/lib.rs +102 -9
  171. package/rust/main.rs +156 -45
  172. package/rust/types/Cargo.toml +8 -0
  173. package/rust/types/src/addons.rs +55 -0
  174. package/rust/types/src/ast.rs +198 -0
  175. package/rust/types/src/config.rs +74 -0
  176. package/rust/types/src/lib.rs +12 -0
  177. package/rust/types/src/telemetry.rs +85 -0
  178. package/rust/utils/Cargo.toml +23 -0
  179. package/rust/utils/{error.rs → src/error.rs} +186 -200
  180. package/rust/utils/src/file.rs +94 -0
  181. package/rust/utils/src/first_usage.rs +97 -0
  182. package/rust/utils/{mod.rs → src/lib.rs} +1 -1
  183. package/rust/utils/{logger.rs → src/logger.rs} +17 -12
  184. package/rust/utils/src/path.rs +88 -0
  185. package/rust/utils/src/signature.rs +41 -0
  186. package/rust/utils/{spinner.rs → src/spinner.rs} +3 -5
  187. package/rust/utils/src/version.rs +27 -0
  188. package/rust/utils/{watcher.rs → src/watcher.rs} +13 -1
  189. package/rust/web/api.rs +5 -0
  190. package/rust/web/cdn.rs +34 -0
  191. package/templates/minimal/README.md +98 -54
  192. package/templates/welcome/README.md +98 -54
  193. package/templates/welcome/src/index.deva +56 -8
  194. package/templates/welcome/src/variables.deva +2 -4
  195. package/tests/rust/TODO.md +0 -0
  196. package/tests/typescript/index.spec.ts +136 -0
  197. package/tests/typescript/playhead.spec.ts +36 -0
  198. package/tests/typescript/render_e2e.spec.ts +77 -0
  199. package/tsconfig.json +1 -1
  200. package/typescript/core/functions/index.ts +83 -0
  201. package/typescript/core/index.ts +6 -0
  202. package/typescript/core/types/index.ts +4 -0
  203. package/typescript/core/types/plugin.ts +19 -0
  204. package/typescript/core/types/result.ts +29 -0
  205. package/typescript/core/types/statement.ts +47 -0
  206. package/typescript/core/types/value.ts +29 -0
  207. package/typescript/index.ts +7 -2
  208. package/typescript/pkg/devalang_core.d.ts +4 -0
  209. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  210. package/typescript/scripts/postinstall.ts +45 -32
  211. package/rust/cli/bank.rs +0 -462
  212. package/rust/cli/build.rs +0 -252
  213. package/rust/cli/generator.rs +0 -1
  214. package/rust/cli/play.rs +0 -1123
  215. package/rust/cli/telemetry.rs +0 -19
  216. package/rust/common/api.rs +0 -5
  217. package/rust/common/cdn.rs +0 -5
  218. package/rust/config/loader.rs +0 -165
  219. package/rust/config/stats.rs +0 -257
  220. package/rust/core/audio/engine.rs +0 -696
  221. package/rust/core/shared/bank.rs +0 -21
  222. package/rust/core/shared/duration.rs +0 -9
  223. package/rust/core/shared/mod.rs +0 -3
  224. package/rust/core/shared/value.rs +0 -35
  225. package/rust/core/utils/validation.rs +0 -35
  226. package/rust/installer/bank.rs +0 -62
  227. package/rust/installer/plugin.rs +0 -54
  228. package/rust/installer/utils.rs +0 -56
  229. package/rust/utils/file.rs +0 -38
  230. package/rust/utils/first_usage.rs +0 -76
  231. package/rust/utils/signature.rs +0 -19
  232. package/rust/utils/telemetry.rs +0 -292
  233. package/rust/utils/version.rs +0 -15
  234. /package/rust/{common → web}/mod.rs +0 -0
  235. /package/rust/{common → web}/sso.rs +0 -0
@@ -1,200 +1,186 @@
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
+ use super::logger::{LogLevel, Logger};
2
+ use devalang_types::{ErrorResult, Severity, StackFrame, Statement, StatementKind, Value};
3
+ use std::collections::HashMap;
4
+
5
+ pub fn collect_errors_recursively(statements: &[Statement]) -> Vec<ErrorResult> {
6
+ let mut errors: Vec<ErrorResult> = Vec::new();
7
+
8
+ for stmt in statements {
9
+ match &stmt.kind {
10
+ StatementKind::Unknown => {
11
+ errors.push(ErrorResult {
12
+ message: format!("Unknown statement at line {}:{}", stmt.line, stmt.column),
13
+ line: stmt.line,
14
+ column: stmt.column,
15
+ severity: Severity::Warning,
16
+ stack: vec![StackFrame {
17
+ module: None,
18
+ context: Some("Unknown".to_string()),
19
+ line: stmt.line,
20
+ column: stmt.column,
21
+ }],
22
+ });
23
+ }
24
+ StatementKind::Error { message } => {
25
+ errors.push(ErrorResult {
26
+ message: message.clone(),
27
+ line: stmt.line,
28
+ column: stmt.column,
29
+ severity: Severity::Critical,
30
+ stack: vec![StackFrame {
31
+ module: None,
32
+ context: Some("Error".to_string()),
33
+ line: stmt.line,
34
+ column: stmt.column,
35
+ }],
36
+ });
37
+ }
38
+ StatementKind::Loop => {
39
+ if let Some(body_statements) = extract_loop_body_statements(&stmt.value) {
40
+ let nested = collect_errors_recursively(body_statements);
41
+ errors.extend(nested.into_iter().map(|mut e| {
42
+ e.stack.insert(
43
+ 0,
44
+ StackFrame {
45
+ module: None,
46
+ context: Some("loop".to_string()),
47
+ line: stmt.line,
48
+ column: stmt.column,
49
+ },
50
+ );
51
+ e
52
+ }));
53
+ }
54
+ }
55
+ _ => {}
56
+ }
57
+ }
58
+
59
+ errors
60
+ }
61
+
62
+ fn extract_loop_body_statements(value: &Value) -> Option<&[Statement]> {
63
+ if let Value::Map(map) = value {
64
+ if let Some(Value::Block(statements)) = map.get("body") {
65
+ return Some(statements);
66
+ }
67
+ }
68
+ None
69
+ }
70
+
71
+ pub fn partition_errors(errors: Vec<ErrorResult>) -> (Vec<ErrorResult>, Vec<ErrorResult>) {
72
+ let mut warnings = Vec::new();
73
+ let mut criticals = Vec::new();
74
+ for e in errors {
75
+ match e.severity {
76
+ Severity::Warning => warnings.push(e),
77
+ Severity::Critical => criticals.push(e),
78
+ }
79
+ }
80
+ (warnings, criticals)
81
+ }
82
+
83
+ pub fn log_errors_with_stack(prefix: &str, warnings: &[ErrorResult], criticals: &[ErrorResult]) {
84
+ let logger = Logger::new();
85
+ if !warnings.is_empty() {
86
+ logger.log_message(
87
+ LogLevel::Warning,
88
+ &format!("{}: {} warning(s)", prefix, warnings.len()),
89
+ );
90
+ for w in warnings {
91
+ logger.log_message(LogLevel::Warning, &format!("- {}", w.message));
92
+ if let Some(frame) = w.stack.first() {
93
+ let module = frame.module.clone().unwrap_or_default();
94
+ logger.log_message(
95
+ LogLevel::Debug,
96
+ &format!(
97
+ " ↳ {}:{}:{} {}",
98
+ module,
99
+ frame.line,
100
+ frame.column,
101
+ frame.context.clone().unwrap_or_default()
102
+ ),
103
+ );
104
+ }
105
+ if w.stack.len() > 1 {
106
+ for (i, f) in w.stack.iter().enumerate().skip(1) {
107
+ let module = f.module.clone().unwrap_or_default();
108
+ logger.log_message(
109
+ LogLevel::Debug,
110
+ &format!(
111
+ " #{} {}:{}:{} {}",
112
+ i,
113
+ module,
114
+ f.line,
115
+ f.column,
116
+ f.context.clone().unwrap_or_default()
117
+ ),
118
+ );
119
+ }
120
+ }
121
+ }
122
+ }
123
+ if !criticals.is_empty() {
124
+ logger.log_message(
125
+ LogLevel::Error,
126
+ &format!("{}: {} critical error(s)", prefix, criticals.len()),
127
+ );
128
+ for c in criticals {
129
+ logger.log_message(LogLevel::Error, &format!("- {}", c.message));
130
+ if let Some(frame) = c.stack.first() {
131
+ let module = frame.module.clone().unwrap_or_default();
132
+ logger.log_message(
133
+ LogLevel::Error,
134
+ &format!(
135
+ " ↳ {}:{}:{} {}",
136
+ module,
137
+ frame.line,
138
+ frame.column,
139
+ frame.context.clone().unwrap_or_default()
140
+ ),
141
+ );
142
+ }
143
+ if c.stack.len() > 1 {
144
+ for (i, f) in c.stack.iter().enumerate().skip(1) {
145
+ let module = f.module.clone().unwrap_or_default();
146
+ logger.log_message(
147
+ LogLevel::Error,
148
+ &format!(
149
+ " #{} {}:{}:{} {}",
150
+ i,
151
+ module,
152
+ f.line,
153
+ f.column,
154
+ f.context.clone().unwrap_or_default()
155
+ ),
156
+ );
157
+ }
158
+ }
159
+ }
160
+ }
161
+ }
162
+
163
+ pub fn collect_all_errors_with_modules(
164
+ modules: &HashMap<String, Vec<Statement>>,
165
+ ) -> Vec<ErrorResult> {
166
+ let mut all = Vec::new();
167
+ for (module_path, stmts) in modules {
168
+ let mut errs = collect_errors_recursively(stmts);
169
+ for e in errs.iter_mut() {
170
+ if e.stack.is_empty() {
171
+ e.stack.push(StackFrame {
172
+ module: Some(module_path.clone()),
173
+ context: None,
174
+ line: e.line,
175
+ column: e.column,
176
+ });
177
+ } else {
178
+ if e.stack[0].module.is_none() {
179
+ e.stack[0].module = Some(module_path.clone());
180
+ }
181
+ }
182
+ }
183
+ all.extend(errs);
184
+ }
185
+ all
186
+ }
@@ -0,0 +1,94 @@
1
+ use include_dir::{Dir, DirEntry};
2
+ use std::{fs, path::Path};
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
+ // Compute the destination path relative to the provided base.
12
+ let rel_path = match file.path().strip_prefix(base_path) {
13
+ Ok(p) => p.to_owned(),
14
+ Err(_) => {
15
+ eprintln!(
16
+ "Warning: failed to compute relative path for {:?}, skipping",
17
+ file.path()
18
+ );
19
+ continue;
20
+ }
21
+ };
22
+
23
+ let dest_path = target_root.join(rel_path);
24
+
25
+ if let Some(parent) = dest_path.parent() {
26
+ if let Err(e) = fs::create_dir_all(parent) {
27
+ eprintln!(
28
+ "Warning: failed to create directory {}: {}",
29
+ parent.display(),
30
+ e
31
+ );
32
+ continue;
33
+ }
34
+ }
35
+
36
+ if let Err(e) = fs::write(&dest_path, file.contents()) {
37
+ eprintln!("Warning: failed to write {}: {}", dest_path.display(), e);
38
+ continue;
39
+ }
40
+ }
41
+ }
42
+ }
43
+ }
44
+
45
+ pub fn format_file_size(bytes: u64) -> String {
46
+ const KB: u64 = 1024;
47
+ const MB: u64 = 1024 * 1024;
48
+
49
+ if bytes >= MB {
50
+ format!("{:.2} Mb", (bytes as f64) / (MB as f64))
51
+ } else if bytes >= KB {
52
+ format!("{:.2} Kb", (bytes as f64) / (KB as f64))
53
+ } else {
54
+ format!("{} bytes", bytes)
55
+ }
56
+ }
57
+
58
+ pub fn extract_zip_safely(archive_path: &Path, dest: &Path) -> Result<(), String> {
59
+ let file = std::fs::File::open(archive_path)
60
+ .map_err(|e| format!("Failed to open archive {}: {}", archive_path.display(), e))?;
61
+ let mut archive = zip::ZipArchive::new(file)
62
+ .map_err(|e| format!("Failed to read archive {}: {}", archive_path.display(), e))?;
63
+
64
+ for i in 0..archive.len() {
65
+ let mut file = archive
66
+ .by_index(i)
67
+ .map_err(|e| format!("Failed to access archive entry {}: {}", i, e))?;
68
+
69
+ let enclosed = match file.enclosed_name() {
70
+ Some(path) => path.to_owned(),
71
+ None => {
72
+ continue;
73
+ }
74
+ };
75
+
76
+ let outpath = dest.join(enclosed);
77
+
78
+ if file.name().ends_with('/') || file.is_dir() {
79
+ std::fs::create_dir_all(&outpath)
80
+ .map_err(|e| format!("Failed to create dir {}: {}", outpath.display(), e))?;
81
+ } else {
82
+ if let Some(p) = outpath.parent() {
83
+ std::fs::create_dir_all(p)
84
+ .map_err(|e| format!("Failed to create parent {}: {}", p.display(), e))?;
85
+ }
86
+ let mut outfile = std::fs::File::create(&outpath)
87
+ .map_err(|e| format!("Failed to create file {}: {}", outpath.display(), e))?;
88
+ std::io::copy(&mut file, &mut outfile)
89
+ .map_err(|e| format!("Failed to write file {}: {}", outpath.display(), e))?;
90
+ }
91
+ }
92
+
93
+ Ok(())
94
+ }
@@ -0,0 +1,97 @@
1
+ #[cfg(feature = "cli")]
2
+ use crossterm::style::{Attribute, SetAttribute};
3
+
4
+ #[cfg(feature = "cli")]
5
+ use std::fmt::Write;
6
+
7
+ use crate::logger::{LogLevel, Logger};
8
+ use crate::signature::get_signature;
9
+ use crate::version::get_version;
10
+ use std::env;
11
+ use std::path::PathBuf;
12
+
13
+ fn get_devalang_homedir() -> PathBuf {
14
+ // Prefer explicit env var, then HOME/USERPROFILE, fallback to current dir
15
+ if let Ok(p) = env::var("DEVALANG_HOME") {
16
+ return PathBuf::from(p);
17
+ }
18
+
19
+ if let Ok(p) = env::var("HOME") {
20
+ return PathBuf::from(p).join(".devalang");
21
+ }
22
+
23
+ if let Ok(p) = env::var("USERPROFILE") {
24
+ return PathBuf::from(p).join(".devalang");
25
+ }
26
+
27
+ env::current_dir()
28
+ .unwrap_or_else(|_| PathBuf::from("."))
29
+ .join(".devalang")
30
+ }
31
+
32
+ pub fn check_is_first_usage() -> bool {
33
+ if get_devalang_homedir().exists() == true {
34
+ false
35
+ } else {
36
+ first_usage_welcome();
37
+ true
38
+ }
39
+ }
40
+
41
+ pub fn first_usage_welcome() {
42
+ std::fs::create_dir_all(get_devalang_homedir()).ok();
43
+
44
+ let version = get_version();
45
+ print!("{}", get_signature(&version));
46
+
47
+ let homedir = get_devalang_homedir().display().to_string();
48
+
49
+ let welcome_msg = format!(
50
+ "Welcome to Devalang ! \n\
51
+ It looks like this is your first time using the tool.\n\
52
+ A configuration file will be created in your home directory.\n\
53
+ (location: '{}')",
54
+ homedir
55
+ );
56
+
57
+ #[cfg(feature = "cli")]
58
+ let mut s = String::new();
59
+ #[cfg(feature = "cli")]
60
+ {
61
+ write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
62
+ write!(&mut s, "{}", welcome_msg).unwrap();
63
+ write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
64
+
65
+ println!("");
66
+ println!("{}", s);
67
+ println!("");
68
+ }
69
+
70
+ #[cfg(not(feature = "cli"))]
71
+ {
72
+ // Fallback: plain output on non-cli (wasm) builds
73
+ println!("");
74
+ println!("{}", welcome_msg);
75
+ println!("");
76
+ }
77
+
78
+ first_usage_ask_for_telemetry();
79
+ }
80
+
81
+ pub fn first_usage_ask_for_telemetry() {
82
+ let telemetry_msg = "Would you like to enable anonymous telemetry ?";
83
+ let telemetry_desc = "This data helps us improve the tool. You can opt-out at any time.";
84
+
85
+ // Non-interactive fallback for first usage: default to telemetry disabled.
86
+ let _ = telemetry_msg;
87
+ let _ = telemetry_desc;
88
+
89
+ let logger = Logger::new();
90
+
91
+ println!("");
92
+ logger.log_message(
93
+ LogLevel::Info,
94
+ "Telemetry disabled by default. You can enable it at any time by using 'devalang telemetry enable'"
95
+ );
96
+ println!("");
97
+ }
@@ -2,8 +2,8 @@ pub mod error;
2
2
  pub mod file;
3
3
  pub mod first_usage;
4
4
  pub mod logger;
5
+ pub mod path;
5
6
  pub mod signature;
6
7
  pub mod spinner;
7
- pub mod telemetry;
8
8
  pub mod version;
9
9
  pub mod watcher;