@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,269 +1,269 @@
1
- use crate::core::parser::{
2
- driver::Parser,
3
- statement::{Statement, StatementKind},
4
- };
5
- use devalang_types::Value;
6
- use serde::{Deserialize, Serialize};
7
-
8
- pub struct ErrorHandler {
9
- errors: Vec<Error>,
10
- }
11
-
12
- #[derive(Serialize, Deserialize, Clone, Debug)]
13
- pub enum Severity {
14
- Warning,
15
- Critical,
16
- }
17
-
18
- #[derive(Serialize, Deserialize, Clone, Debug, Default)]
19
- pub struct StackFrame {
20
- pub module: Option<String>,
21
- pub context: Option<String>,
22
- pub line: usize,
23
- pub column: usize,
24
- }
25
-
26
- #[derive(Serialize, Deserialize, Clone, Debug)]
27
- pub struct ErrorResult {
28
- pub message: String,
29
- pub line: usize,
30
- pub column: usize,
31
- pub severity: Severity,
32
- pub stack: Vec<StackFrame>,
33
- }
34
-
35
- #[derive(Clone)]
36
- pub struct Error {
37
- pub message: String,
38
- pub line: usize,
39
- pub column: usize,
40
- }
41
-
42
- impl Default for ErrorHandler {
43
- fn default() -> Self {
44
- Self::new()
45
- }
46
- }
47
-
48
- impl ErrorHandler {
49
- pub fn new() -> Self {
50
- Self { errors: Vec::new() }
51
- }
52
-
53
- pub fn add_error(&mut self, message: String, line: usize, column: usize) {
54
- let error_statement = Error {
55
- message,
56
- line,
57
- column,
58
- };
59
- self.errors.push(error_statement);
60
- }
61
-
62
- pub fn has_errors(&self) -> bool {
63
- !self.errors.is_empty()
64
- }
65
-
66
- pub fn get_errors(&self) -> &Vec<Error> {
67
- &self.errors
68
- }
69
-
70
- pub fn clear_errors(&mut self) {
71
- self.errors.clear();
72
- }
73
-
74
- pub fn detect_from_statements(&mut self, _parser: &mut Parser, statements: &[Statement]) {
75
- for stmt in statements {
76
- match &stmt.kind {
77
- StatementKind::Unknown => {
78
- self.add_error("Unknown statement".to_string(), stmt.line, stmt.column);
79
- }
80
- StatementKind::Error { message } => {
81
- self.add_error(message.clone(), stmt.line, stmt.column);
82
- }
83
- _ => {}
84
- }
85
- }
86
- }
87
- }
88
-
89
- /// Collects errors recursively from statements (mirrors old utils implementation).
90
- pub fn collect_errors_recursively(statements: &[Statement]) -> Vec<ErrorResult> {
91
- let mut errors: Vec<ErrorResult> = Vec::new();
92
-
93
- for stmt in statements {
94
- match &stmt.kind {
95
- StatementKind::Unknown => {
96
- errors.push(ErrorResult {
97
- message: format!("Unknown statement at line {}:{}", stmt.line, stmt.column),
98
- line: stmt.line,
99
- column: stmt.column,
100
- severity: Severity::Warning,
101
- stack: vec![StackFrame {
102
- module: None,
103
- context: Some("Unknown".to_string()),
104
- line: stmt.line,
105
- column: stmt.column,
106
- }],
107
- });
108
- }
109
- StatementKind::Error { message } => {
110
- errors.push(ErrorResult {
111
- message: message.clone(),
112
- line: stmt.line,
113
- column: stmt.column,
114
- severity: Severity::Critical,
115
- stack: vec![StackFrame {
116
- module: None,
117
- context: Some("Error".to_string()),
118
- line: stmt.line,
119
- column: stmt.column,
120
- }],
121
- });
122
- }
123
- StatementKind::Loop => {
124
- if let Some(body_statements) = extract_loop_body_statements(&stmt.value) {
125
- let nested = collect_errors_recursively(body_statements);
126
- errors.extend(nested.into_iter().map(|mut e| {
127
- e.stack.insert(
128
- 0,
129
- StackFrame {
130
- module: None,
131
- context: Some("loop".to_string()),
132
- line: stmt.line,
133
- column: stmt.column,
134
- },
135
- );
136
- e
137
- }));
138
- }
139
- }
140
- _ => {}
141
- }
142
- }
143
-
144
- errors
145
- }
146
-
147
- fn extract_loop_body_statements(value: &Value) -> Option<&[Statement]> {
148
- if let Value::Map(map) = value {
149
- if let Some(Value::Block(statements)) = map.get("body") {
150
- return Some(statements);
151
- }
152
- }
153
- None
154
- }
155
-
156
- pub fn partition_errors(errors: Vec<ErrorResult>) -> (Vec<ErrorResult>, Vec<ErrorResult>) {
157
- let mut warnings = Vec::new();
158
- let mut criticals = Vec::new();
159
- for e in errors {
160
- match e.severity {
161
- Severity::Warning => warnings.push(e),
162
- Severity::Critical => criticals.push(e),
163
- }
164
- }
165
- (warnings, criticals)
166
- }
167
-
168
- pub fn log_errors_with_stack(prefix: &str, warnings: &[ErrorResult], criticals: &[ErrorResult]) {
169
- use devalang_utils::logger::LogLevel;
170
- use devalang_utils::logger::Logger;
171
-
172
- let logger = Logger::new();
173
- if !warnings.is_empty() {
174
- logger.log_message(
175
- LogLevel::Warning,
176
- &format!("{}: {} warning(s)", prefix, warnings.len()),
177
- );
178
- for w in warnings {
179
- logger.log_message(LogLevel::Warning, &format!("- {}", w.message));
180
- if let Some(frame) = w.stack.first() {
181
- let module = frame.module.clone().unwrap_or_default();
182
- logger.log_message(
183
- LogLevel::Debug,
184
- &format!(
185
- " ↳ {}:{}:{} {}",
186
- module,
187
- frame.line,
188
- frame.column,
189
- frame.context.clone().unwrap_or_default()
190
- ),
191
- );
192
- }
193
- if w.stack.len() > 1 {
194
- for (i, f) in w.stack.iter().enumerate().skip(1) {
195
- let module = f.module.clone().unwrap_or_default();
196
- logger.log_message(
197
- LogLevel::Debug,
198
- &format!(
199
- " #{} {}:{}:{} {}",
200
- i,
201
- module,
202
- f.line,
203
- f.column,
204
- f.context.clone().unwrap_or_default()
205
- ),
206
- );
207
- }
208
- }
209
- }
210
- }
211
- if !criticals.is_empty() {
212
- logger.log_message(
213
- LogLevel::Error,
214
- &format!("{}: {} critical error(s)", prefix, criticals.len()),
215
- );
216
- for c in criticals {
217
- logger.log_message(LogLevel::Error, &format!("- {}", c.message));
218
- if let Some(frame) = c.stack.first() {
219
- let module = frame.module.clone().unwrap_or_default();
220
- logger.log_message(
221
- LogLevel::Error,
222
- &format!(
223
- " ↳ {}:{}:{} {}",
224
- module,
225
- frame.line,
226
- frame.column,
227
- frame.context.clone().unwrap_or_default()
228
- ),
229
- );
230
- }
231
- if c.stack.len() > 1 {
232
- for (i, f) in c.stack.iter().enumerate().skip(1) {
233
- let module = f.module.clone().unwrap_or_default();
234
- logger.log_message(
235
- LogLevel::Error,
236
- &format!(
237
- " #{} {}:{}:{} {}",
238
- i,
239
- module,
240
- f.line,
241
- f.column,
242
- f.context.clone().unwrap_or_default()
243
- ),
244
- );
245
- }
246
- }
247
- }
248
- }
249
- }
250
-
251
- /// Collects errors from all modules and annotates stack frames with module names.
252
- pub fn collect_all_errors_with_modules(
253
- statements_by_module: &std::collections::HashMap<String, Vec<Statement>>,
254
- ) -> Vec<ErrorResult> {
255
- let mut all: Vec<ErrorResult> = Vec::new();
256
- for (module, stmts) in statements_by_module.iter() {
257
- let mut errs = collect_errors_recursively(stmts);
258
- for e in errs.iter_mut() {
259
- // annotate first stack frame module if missing
260
- if let Some(first) = e.stack.first_mut() {
261
- if first.module.is_none() {
262
- first.module = Some(module.clone());
263
- }
264
- }
265
- }
266
- all.extend(errs.into_iter());
267
- }
268
- all
269
- }
1
+ use crate::core::parser::{
2
+ driver::parser::Parser,
3
+ statement::{Statement, StatementKind},
4
+ };
5
+ use devalang_types::Value;
6
+ use serde::{Deserialize, Serialize};
7
+
8
+ pub struct ErrorHandler {
9
+ errors: Vec<Error>,
10
+ }
11
+
12
+ #[derive(Serialize, Deserialize, Clone, Debug)]
13
+ pub enum Severity {
14
+ Warning,
15
+ Critical,
16
+ }
17
+
18
+ #[derive(Serialize, Deserialize, Clone, Debug, Default)]
19
+ pub struct StackFrame {
20
+ pub module: Option<String>,
21
+ pub context: Option<String>,
22
+ pub line: usize,
23
+ pub column: usize,
24
+ }
25
+
26
+ #[derive(Serialize, Deserialize, Clone, Debug)]
27
+ pub struct ErrorResult {
28
+ pub message: String,
29
+ pub line: usize,
30
+ pub column: usize,
31
+ pub severity: Severity,
32
+ pub stack: Vec<StackFrame>,
33
+ }
34
+
35
+ #[derive(Clone)]
36
+ pub struct Error {
37
+ pub message: String,
38
+ pub line: usize,
39
+ pub column: usize,
40
+ }
41
+
42
+ impl Default for ErrorHandler {
43
+ fn default() -> Self {
44
+ Self::new()
45
+ }
46
+ }
47
+
48
+ impl ErrorHandler {
49
+ pub fn new() -> Self {
50
+ Self { errors: Vec::new() }
51
+ }
52
+
53
+ pub fn add_error(&mut self, message: String, line: usize, column: usize) {
54
+ let error_statement = Error {
55
+ message,
56
+ line,
57
+ column,
58
+ };
59
+ self.errors.push(error_statement);
60
+ }
61
+
62
+ pub fn has_errors(&self) -> bool {
63
+ !self.errors.is_empty()
64
+ }
65
+
66
+ pub fn get_errors(&self) -> &Vec<Error> {
67
+ &self.errors
68
+ }
69
+
70
+ pub fn clear_errors(&mut self) {
71
+ self.errors.clear();
72
+ }
73
+
74
+ pub fn detect_from_statements(&mut self, _parser: &mut Parser, statements: &[Statement]) {
75
+ for stmt in statements {
76
+ match &stmt.kind {
77
+ StatementKind::Unknown => {
78
+ self.add_error("Unknown statement".to_string(), stmt.line, stmt.column);
79
+ }
80
+ StatementKind::Error { message } => {
81
+ self.add_error(message.clone(), stmt.line, stmt.column);
82
+ }
83
+ _ => {}
84
+ }
85
+ }
86
+ }
87
+ }
88
+
89
+ /// Collects errors recursively from statements (mirrors old utils implementation).
90
+ pub fn collect_errors_recursively(statements: &[Statement]) -> Vec<ErrorResult> {
91
+ let mut errors: Vec<ErrorResult> = Vec::new();
92
+
93
+ for stmt in statements {
94
+ match &stmt.kind {
95
+ StatementKind::Unknown => {
96
+ errors.push(ErrorResult {
97
+ message: format!("Unknown statement at line {}:{}", stmt.line, stmt.column),
98
+ line: stmt.line,
99
+ column: stmt.column,
100
+ severity: Severity::Warning,
101
+ stack: vec![StackFrame {
102
+ module: None,
103
+ context: Some("Unknown".to_string()),
104
+ line: stmt.line,
105
+ column: stmt.column,
106
+ }],
107
+ });
108
+ }
109
+ StatementKind::Error { message } => {
110
+ errors.push(ErrorResult {
111
+ message: message.clone(),
112
+ line: stmt.line,
113
+ column: stmt.column,
114
+ severity: Severity::Critical,
115
+ stack: vec![StackFrame {
116
+ module: None,
117
+ context: Some("Error".to_string()),
118
+ line: stmt.line,
119
+ column: stmt.column,
120
+ }],
121
+ });
122
+ }
123
+ StatementKind::Loop => {
124
+ if let Some(body_statements) = extract_loop_body_statements(&stmt.value) {
125
+ let nested = collect_errors_recursively(body_statements);
126
+ errors.extend(nested.into_iter().map(|mut e| {
127
+ e.stack.insert(
128
+ 0,
129
+ StackFrame {
130
+ module: None,
131
+ context: Some("loop".to_string()),
132
+ line: stmt.line,
133
+ column: stmt.column,
134
+ },
135
+ );
136
+ e
137
+ }));
138
+ }
139
+ }
140
+ _ => {}
141
+ }
142
+ }
143
+
144
+ errors
145
+ }
146
+
147
+ fn extract_loop_body_statements(value: &Value) -> Option<&[Statement]> {
148
+ if let Value::Map(map) = value {
149
+ if let Some(Value::Block(statements)) = map.get("body") {
150
+ return Some(statements);
151
+ }
152
+ }
153
+ None
154
+ }
155
+
156
+ pub fn partition_errors(errors: Vec<ErrorResult>) -> (Vec<ErrorResult>, Vec<ErrorResult>) {
157
+ let mut warnings = Vec::new();
158
+ let mut criticals = Vec::new();
159
+ for e in errors {
160
+ match e.severity {
161
+ Severity::Warning => warnings.push(e),
162
+ Severity::Critical => criticals.push(e),
163
+ }
164
+ }
165
+ (warnings, criticals)
166
+ }
167
+
168
+ pub fn log_errors_with_stack(prefix: &str, warnings: &[ErrorResult], criticals: &[ErrorResult]) {
169
+ use devalang_utils::logger::LogLevel;
170
+ use devalang_utils::logger::Logger;
171
+
172
+ let logger = Logger::new();
173
+ if !warnings.is_empty() {
174
+ logger.log_message(
175
+ LogLevel::Warning,
176
+ &format!("{}: {} warning(s)", prefix, warnings.len()),
177
+ );
178
+ for w in warnings {
179
+ logger.log_message(LogLevel::Warning, &format!("- {}", w.message));
180
+ if let Some(frame) = w.stack.first() {
181
+ let module = frame.module.clone().unwrap_or_default();
182
+ logger.log_message(
183
+ LogLevel::Debug,
184
+ &format!(
185
+ " ↳ {}:{}:{} {}",
186
+ module,
187
+ frame.line,
188
+ frame.column,
189
+ frame.context.clone().unwrap_or_default()
190
+ ),
191
+ );
192
+ }
193
+ if w.stack.len() > 1 {
194
+ for (i, f) in w.stack.iter().enumerate().skip(1) {
195
+ let module = f.module.clone().unwrap_or_default();
196
+ logger.log_message(
197
+ LogLevel::Debug,
198
+ &format!(
199
+ " #{} {}:{}:{} {}",
200
+ i,
201
+ module,
202
+ f.line,
203
+ f.column,
204
+ f.context.clone().unwrap_or_default()
205
+ ),
206
+ );
207
+ }
208
+ }
209
+ }
210
+ }
211
+ if !criticals.is_empty() {
212
+ logger.log_message(
213
+ LogLevel::Error,
214
+ &format!("{}: {} critical error(s)", prefix, criticals.len()),
215
+ );
216
+ for c in criticals {
217
+ logger.log_message(LogLevel::Error, &format!("- {}", c.message));
218
+ if let Some(frame) = c.stack.first() {
219
+ let module = frame.module.clone().unwrap_or_default();
220
+ logger.log_message(
221
+ LogLevel::Error,
222
+ &format!(
223
+ " ↳ {}:{}:{} {}",
224
+ module,
225
+ frame.line,
226
+ frame.column,
227
+ frame.context.clone().unwrap_or_default()
228
+ ),
229
+ );
230
+ }
231
+ if c.stack.len() > 1 {
232
+ for (i, f) in c.stack.iter().enumerate().skip(1) {
233
+ let module = f.module.clone().unwrap_or_default();
234
+ logger.log_message(
235
+ LogLevel::Error,
236
+ &format!(
237
+ " #{} {}:{}:{} {}",
238
+ i,
239
+ module,
240
+ f.line,
241
+ f.column,
242
+ f.context.clone().unwrap_or_default()
243
+ ),
244
+ );
245
+ }
246
+ }
247
+ }
248
+ }
249
+ }
250
+
251
+ /// Collects errors from all modules and annotates stack frames with module names.
252
+ pub fn collect_all_errors_with_modules(
253
+ statements_by_module: &std::collections::HashMap<String, Vec<Statement>>,
254
+ ) -> Vec<ErrorResult> {
255
+ let mut all: Vec<ErrorResult> = Vec::new();
256
+ for (module, stmts) in statements_by_module.iter() {
257
+ let mut errs = collect_errors_recursively(stmts);
258
+ for e in errs.iter_mut() {
259
+ // annotate first stack frame module if missing
260
+ if let Some(first) = e.stack.first_mut() {
261
+ if first.module.is_none() {
262
+ first.module = Some(module.clone());
263
+ }
264
+ }
265
+ }
266
+ all.extend(errs.into_iter());
267
+ }
268
+ all
269
+ }
@@ -1,61 +1,59 @@
1
- use crate::core::{
2
- lexer::{handler::driver::handle_content_lexing, token::Token},
3
- utils::path::normalize_path,
4
- };
5
- use std::fs;
6
- use std::path::Path;
7
-
8
- pub struct Lexer {}
9
-
10
- impl Default for Lexer {
11
- fn default() -> Self {
12
- Self::new()
13
- }
14
- }
15
-
16
- impl Lexer {
17
- pub fn new() -> Self {
18
- Lexer {}
19
- }
20
-
21
- pub fn lex_from_source(&self, source: &str) -> Result<Vec<Token>, String> {
22
- handle_content_lexing(source.to_string())
23
- }
24
-
25
- pub fn lex_tokens(&self, entrypoint: &str) -> Result<Vec<Token>, String> {
26
- let path = normalize_path(entrypoint);
27
- let resolved_path = Self::resolve_entry_path(&path)?;
28
-
29
- let file_content = fs::read_to_string(&resolved_path).map_err(|e| {
30
- format!(
31
- "Failed to read the entrypoint file '{}': {}",
32
- resolved_path, e
33
- )
34
- })?;
35
-
36
- handle_content_lexing(file_content).map_err(|e| format!("Failed to lex the content: {}", e))
37
- }
38
-
39
- fn resolve_entry_path(path: &str) -> Result<String, String> {
40
- let candidate = Path::new(path);
41
-
42
- if candidate.is_dir() {
43
- let index_path = candidate.join("index.deva");
44
- if index_path.exists() {
45
- Ok(index_path.to_string_lossy().replace("\\", "/"))
46
- } else {
47
- Err(format!(
48
- "Expected 'index.deva' in directory '{}', but it was not found",
49
- path
50
- ))
51
- }
52
- } else if candidate.is_file() {
53
- return Ok(path.to_string());
54
- } else {
55
- return Err(format!(
56
- "Provided entrypoint '{}' is not a valid file or directory",
57
- path
58
- ));
59
- }
60
- }
61
- }
1
+ use crate::core::lexer::{handler::driver::handle_content_lexing, token::Token};
2
+ use devalang_utils::path::normalize_path;
3
+ use std::fs;
4
+ use std::path::Path;
5
+
6
+ pub struct Lexer {}
7
+
8
+ impl Default for Lexer {
9
+ fn default() -> Self {
10
+ Self::new()
11
+ }
12
+ }
13
+
14
+ impl Lexer {
15
+ pub fn new() -> Self {
16
+ Lexer {}
17
+ }
18
+
19
+ pub fn lex_from_source(&self, source: &str) -> Result<Vec<Token>, String> {
20
+ handle_content_lexing(source.to_string())
21
+ }
22
+
23
+ pub fn lex_tokens(&self, entrypoint: &str) -> Result<Vec<Token>, String> {
24
+ let path = normalize_path(entrypoint);
25
+ let resolved_path = Self::resolve_entry_path(&path)?;
26
+
27
+ let file_content = fs::read_to_string(&resolved_path).map_err(|e| {
28
+ format!(
29
+ "Failed to read the entrypoint file '{}': {}",
30
+ resolved_path, e
31
+ )
32
+ })?;
33
+
34
+ handle_content_lexing(file_content).map_err(|e| format!("Failed to lex the content: {}", e))
35
+ }
36
+
37
+ fn resolve_entry_path(path: &str) -> Result<String, String> {
38
+ let candidate = Path::new(path);
39
+
40
+ if candidate.is_dir() {
41
+ let index_path = candidate.join("index.deva");
42
+ if index_path.exists() {
43
+ Ok(index_path.to_string_lossy().replace("\\", "/"))
44
+ } else {
45
+ Err(format!(
46
+ "Expected 'index.deva' in directory '{}', but it was not found",
47
+ path
48
+ ))
49
+ }
50
+ } else if candidate.is_file() {
51
+ return Ok(path.to_string());
52
+ } else {
53
+ return Err(format!(
54
+ "Provided entrypoint '{}' is not a valid file or directory",
55
+ path
56
+ ));
57
+ }
58
+ }
59
+ }