@devaloop/devalang 0.0.1-alpha.9 → 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 (322) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +6 -1
  3. package/.github/workflows/ci.yml +103 -0
  4. package/Cargo.toml +81 -48
  5. package/README.md +137 -154
  6. package/docs/CHANGELOG.md +428 -1
  7. package/docs/CONTRIBUTING.md +101 -0
  8. package/docs/ROADMAP.md +14 -7
  9. package/docs/TODO.md +16 -15
  10. package/examples/automation.deva +42 -0
  11. package/examples/bank.deva +7 -0
  12. package/examples/bus.deva +10 -0
  13. package/examples/duration.deva +9 -0
  14. package/examples/effect.deva +2 -0
  15. package/examples/events.deva +12 -0
  16. package/examples/filter.deva +11 -0
  17. package/examples/function.deva +15 -0
  18. package/examples/index.deva +57 -12
  19. package/examples/lfo.deva +9 -0
  20. package/examples/loop.deva +5 -12
  21. package/examples/pattern.deva +8 -0
  22. package/examples/plugin.deva +16 -0
  23. package/examples/synth.deva +11 -1
  24. package/examples/synth_types.deva +17 -0
  25. package/examples/variables.deva +1 -1
  26. package/out-tsc/bin/index.d.ts +2 -0
  27. package/out-tsc/bin/index.js +51 -7
  28. package/out-tsc/core/functions/index.d.ts +42 -0
  29. package/out-tsc/core/functions/index.js +87 -0
  30. package/out-tsc/core/index.d.ts +6 -0
  31. package/out-tsc/core/index.js +22 -0
  32. package/out-tsc/core/types/index.d.ts +4 -0
  33. package/out-tsc/core/types/index.js +20 -0
  34. package/out-tsc/core/types/plugin.d.ts +18 -0
  35. package/out-tsc/core/types/plugin.js +2 -0
  36. package/out-tsc/core/types/result.d.ts +27 -0
  37. package/out-tsc/core/types/result.js +2 -0
  38. package/out-tsc/core/types/statement.d.ts +106 -0
  39. package/out-tsc/core/types/statement.js +2 -0
  40. package/out-tsc/core/types/value.d.ts +43 -0
  41. package/out-tsc/core/types/value.js +2 -0
  42. package/out-tsc/index.d.ts +7 -0
  43. package/out-tsc/index.js +42 -1
  44. package/out-tsc/pkg/devalang_core.d.ts +15 -0
  45. package/out-tsc/pkg/devalang_core.js +65 -0
  46. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +34 -0
  47. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  48. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  49. package/out-tsc/scripts/postinstall.d.ts +1 -0
  50. package/out-tsc/scripts/postinstall.js +83 -0
  51. package/out-tsc/scripts/version/bump.d.ts +1 -0
  52. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  53. package/out-tsc/scripts/version/index.d.ts +1 -0
  54. package/out-tsc/scripts/version/sync.d.ts +1 -0
  55. package/package.json +28 -7
  56. package/project-version.json +4 -4
  57. package/rust/cli/bank/api.rs +122 -0
  58. package/rust/cli/bank/commands.rs +306 -0
  59. package/rust/cli/bank/mod.rs +29 -0
  60. package/rust/cli/build/commands.rs +153 -0
  61. package/rust/cli/build/mod.rs +2 -0
  62. package/rust/cli/build/process.rs +165 -0
  63. package/rust/cli/check/mod.rs +208 -0
  64. package/rust/cli/discover/commands.rs +253 -0
  65. package/rust/cli/discover/config.rs +111 -0
  66. package/rust/cli/discover/fs.rs +19 -0
  67. package/rust/cli/discover/install.rs +103 -0
  68. package/rust/cli/discover/metadata.rs +48 -0
  69. package/rust/cli/discover/mod.rs +5 -0
  70. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  71. package/rust/cli/init/mod.rs +1 -0
  72. package/rust/cli/install/addon.rs +118 -0
  73. package/rust/cli/install/bank.rs +72 -0
  74. package/rust/cli/install/commands.rs +35 -0
  75. package/rust/cli/install/mod.rs +4 -0
  76. package/rust/cli/install/plugin.rs +80 -0
  77. package/rust/cli/login/commands.rs +124 -0
  78. package/rust/cli/login/mod.rs +1 -0
  79. package/rust/cli/mod.rs +9 -202
  80. package/rust/cli/parser.rs +359 -0
  81. package/rust/cli/play/commands.rs +375 -0
  82. package/rust/cli/play/io.rs +17 -0
  83. package/rust/cli/play/mod.rs +5 -0
  84. package/rust/cli/play/process.rs +159 -0
  85. package/rust/cli/play/realtime.rs +91 -0
  86. package/rust/cli/play/utils.rs +23 -0
  87. package/rust/cli/telemetry/commands.rs +22 -0
  88. package/rust/cli/telemetry/event_creator.rs +80 -0
  89. package/rust/cli/telemetry/mod.rs +3 -0
  90. package/rust/cli/telemetry/send.rs +51 -0
  91. package/rust/cli/{template.rs → template/commands.rs} +17 -5
  92. package/rust/cli/template/mod.rs +1 -0
  93. package/rust/cli/update/commands.rs +6 -0
  94. package/rust/cli/update/mod.rs +1 -0
  95. package/rust/config/driver.rs +112 -0
  96. package/rust/config/mod.rs +3 -16
  97. package/rust/config/ops.rs +26 -0
  98. package/rust/config/settings.rs +101 -0
  99. package/rust/core/audio/engine/driver.rs +220 -0
  100. package/rust/core/audio/engine/export.rs +169 -0
  101. package/rust/core/audio/engine/helpers.rs +178 -0
  102. package/rust/core/audio/engine/mod.rs +56 -0
  103. package/rust/core/audio/engine/notes/dsp.rs +85 -0
  104. package/rust/core/audio/engine/notes/mod.rs +44 -0
  105. package/rust/core/audio/engine/notes/params.rs +294 -0
  106. package/rust/core/audio/engine/sample/insert.rs +199 -0
  107. package/rust/core/audio/engine/sample/mod.rs +40 -0
  108. package/rust/core/audio/engine/sample/padding.rs +170 -0
  109. package/rust/core/audio/evaluator/condition.rs +61 -0
  110. package/rust/core/audio/evaluator/mod.rs +9 -0
  111. package/rust/core/audio/evaluator/numeric.rs +152 -0
  112. package/rust/core/audio/evaluator/rhs.rs +16 -0
  113. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  114. package/rust/core/audio/interpreter/driver.rs +574 -216
  115. package/rust/core/audio/interpreter/mod.rs +2 -12
  116. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
  117. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
  118. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
  119. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
  120. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  121. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  122. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  123. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  124. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  125. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  126. package/rust/core/audio/interpreter/statements/automate.rs +16 -0
  127. package/rust/core/audio/interpreter/statements/call.rs +295 -0
  128. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -69
  129. package/rust/core/audio/interpreter/statements/function.rs +24 -0
  130. package/rust/core/audio/interpreter/statements/let_.rs +36 -0
  131. package/rust/core/audio/interpreter/statements/load.rs +17 -0
  132. package/rust/core/audio/interpreter/statements/loop_.rs +115 -0
  133. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  134. package/rust/core/audio/interpreter/statements/sleep.rs +28 -0
  135. package/rust/core/audio/interpreter/statements/spawn.rs +253 -0
  136. package/rust/core/audio/interpreter/statements/tempo.rs +40 -0
  137. package/rust/core/audio/interpreter/statements/trigger.rs +239 -0
  138. package/rust/core/audio/loader/mod.rs +1 -1
  139. package/rust/core/audio/loader/trigger.rs +98 -52
  140. package/rust/core/audio/mod.rs +2 -2
  141. package/rust/core/audio/player.rs +28 -12
  142. package/rust/core/audio/special/easing.rs +189 -0
  143. package/rust/core/audio/special/env.rs +45 -0
  144. package/rust/core/audio/special/math.rs +134 -0
  145. package/rust/core/audio/special/mod.rs +9 -0
  146. package/rust/core/audio/special/modulator.rs +143 -0
  147. package/rust/core/builder/mod.rs +129 -80
  148. package/rust/core/debugger/lexer.rs +4 -4
  149. package/rust/core/debugger/logs.rs +52 -0
  150. package/rust/core/debugger/mod.rs +11 -2
  151. package/rust/core/debugger/preprocessor.rs +4 -4
  152. package/rust/core/debugger/store.rs +38 -25
  153. package/rust/core/error/mod.rs +221 -12
  154. package/rust/core/lexer/driver.rs +59 -0
  155. package/rust/core/lexer/handler/arrow.rs +62 -11
  156. package/rust/core/lexer/handler/at.rs +5 -5
  157. package/rust/core/lexer/handler/brace.rs +11 -11
  158. package/rust/core/lexer/handler/colon.rs +5 -5
  159. package/rust/core/lexer/handler/comment.rs +3 -3
  160. package/rust/core/lexer/handler/dot.rs +6 -6
  161. package/rust/core/lexer/handler/driver.rs +143 -32
  162. package/rust/core/lexer/handler/identifier.rs +11 -5
  163. package/rust/core/lexer/handler/indent.rs +18 -4
  164. package/rust/core/lexer/handler/mod.rs +6 -5
  165. package/rust/core/lexer/handler/newline.rs +3 -3
  166. package/rust/core/lexer/handler/number.rs +5 -5
  167. package/rust/core/lexer/handler/operator.rs +5 -3
  168. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  169. package/rust/core/lexer/handler/slash.rs +21 -0
  170. package/rust/core/lexer/handler/string.rs +3 -3
  171. package/rust/core/lexer/mod.rs +1 -49
  172. package/rust/core/lexer/token.rs +17 -12
  173. package/rust/core/mod.rs +9 -10
  174. package/rust/core/parser/driver/block.rs +111 -0
  175. package/rust/core/parser/driver/cursor.rs +82 -0
  176. package/rust/core/parser/driver/driver_impl.rs +139 -0
  177. package/rust/core/parser/driver/mod.rs +6 -0
  178. package/rust/core/parser/driver/parse_array.rs +120 -0
  179. package/rust/core/parser/driver/parse_map.rs +223 -0
  180. package/rust/core/parser/driver/parser.rs +160 -0
  181. package/rust/core/parser/handler/arrow_call.rs +277 -126
  182. package/rust/core/parser/handler/at.rs +142 -25
  183. package/rust/core/parser/handler/bank.rs +83 -20
  184. package/rust/core/parser/handler/condition.rs +14 -5
  185. package/rust/core/parser/handler/dot.rs +111 -75
  186. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  187. package/rust/core/parser/handler/identifier/call.rs +74 -24
  188. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  189. package/rust/core/parser/handler/identifier/function.rs +113 -0
  190. package/rust/core/parser/handler/identifier/group.rs +28 -14
  191. package/rust/core/parser/handler/identifier/let_.rs +61 -21
  192. package/rust/core/parser/handler/identifier/mod.rs +24 -20
  193. package/rust/core/parser/handler/identifier/on.rs +107 -0
  194. package/rust/core/parser/handler/identifier/print.rs +49 -0
  195. package/rust/core/parser/handler/identifier/sleep.rs +77 -14
  196. package/rust/core/parser/handler/identifier/spawn.rs +81 -31
  197. package/rust/core/parser/handler/identifier/synth.rs +102 -32
  198. package/rust/core/parser/handler/loop_.rs +144 -22
  199. package/rust/core/parser/handler/mod.rs +6 -5
  200. package/rust/core/parser/handler/pattern.rs +74 -0
  201. package/rust/core/parser/handler/tempo.rs +67 -9
  202. package/rust/core/parser/mod.rs +3 -4
  203. package/rust/core/parser/statement.rs +6 -92
  204. package/rust/core/plugin/loader.rs +137 -0
  205. package/rust/core/plugin/mod.rs +2 -0
  206. package/rust/core/plugin/runner/mod.rs +11 -0
  207. package/rust/core/plugin/runner/non_wasm.rs +297 -0
  208. package/rust/core/plugin/runner/wasm32.rs +43 -0
  209. package/rust/core/preprocessor/loader/inject.rs +278 -0
  210. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  211. package/rust/core/preprocessor/loader/mod.rs +235 -0
  212. package/rust/core/preprocessor/mod.rs +4 -4
  213. package/rust/core/preprocessor/module.rs +55 -50
  214. package/rust/core/preprocessor/processor/handlers.rs +107 -0
  215. package/rust/core/preprocessor/processor/mod.rs +1 -0
  216. package/rust/core/preprocessor/resolver/bank.rs +14 -12
  217. package/rust/core/preprocessor/resolver/call.rs +106 -105
  218. package/rust/core/preprocessor/resolver/condition.rs +13 -10
  219. package/rust/core/preprocessor/resolver/driver.rs +145 -48
  220. package/rust/core/preprocessor/resolver/function.rs +69 -0
  221. package/rust/core/preprocessor/resolver/group.rs +122 -61
  222. package/rust/core/preprocessor/resolver/let_.rs +13 -12
  223. package/rust/core/preprocessor/resolver/loop_.rs +240 -13
  224. package/rust/core/preprocessor/resolver/mod.rs +8 -6
  225. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  226. package/rust/core/preprocessor/resolver/spawn.rs +83 -42
  227. package/rust/core/preprocessor/resolver/synth.rs +15 -11
  228. package/rust/core/preprocessor/resolver/tempo.rs +13 -14
  229. package/rust/core/preprocessor/resolver/trigger.rs +32 -28
  230. package/rust/core/preprocessor/resolver/value.rs +111 -13
  231. package/rust/core/store/global.rs +57 -39
  232. package/rust/core/store/mod.rs +0 -3
  233. package/rust/lib.rs +323 -117
  234. package/rust/main.rs +388 -65
  235. package/rust/types/Cargo.toml +11 -0
  236. package/rust/types/src/addons.rs +55 -0
  237. package/rust/types/src/ast.rs +202 -0
  238. package/rust/types/src/config.rs +84 -0
  239. package/rust/types/src/lib.rs +15 -0
  240. package/rust/types/src/plugin.rs +20 -0
  241. package/rust/types/src/store.rs +139 -0
  242. package/rust/types/src/telemetry.rs +85 -0
  243. package/rust/utils/Cargo.toml +26 -0
  244. package/rust/utils/src/error.rs +186 -0
  245. package/rust/utils/src/file.rs +94 -0
  246. package/rust/utils/src/first_usage.rs +97 -0
  247. package/rust/utils/{mod.rs → src/lib.rs} +6 -3
  248. package/rust/utils/{logger.rs → src/logger.rs} +94 -17
  249. package/rust/utils/src/path.rs +129 -0
  250. package/rust/utils/src/signature.rs +41 -0
  251. package/rust/utils/{spinner.rs → src/spinner.rs} +7 -8
  252. package/rust/utils/src/version.rs +27 -0
  253. package/rust/utils/{watcher.rs → src/watcher.rs} +17 -4
  254. package/rust/web/api.rs +5 -0
  255. package/rust/web/cdn.rs +34 -0
  256. package/rust/web/mod.rs +3 -0
  257. package/rust/web/sso.rs +5 -0
  258. package/templates/minimal/README.md +143 -127
  259. package/templates/welcome/README.md +143 -127
  260. package/templates/welcome/src/index.deva +56 -8
  261. package/templates/welcome/src/variables.deva +2 -4
  262. package/tests/integration.rs +21 -0
  263. package/tests/rust/cli_check_build.rs +21 -0
  264. package/tests/rust/cli_help.rs +12 -0
  265. package/tests/rust/cli_template_list.rs +10 -0
  266. package/tests/rust/cli_version.rs +11 -0
  267. package/tests/typescript/index.spec.ts +136 -0
  268. package/tests/typescript/playhead.spec.ts +36 -0
  269. package/tests/typescript/render_e2e.spec.ts +77 -0
  270. package/tsconfig.json +12 -10
  271. package/typescript/bin/index.ts +19 -5
  272. package/typescript/core/functions/index.ts +94 -0
  273. package/typescript/core/index.ts +6 -0
  274. package/typescript/core/types/index.ts +4 -0
  275. package/typescript/core/types/plugin.ts +19 -0
  276. package/typescript/core/types/result.ts +29 -0
  277. package/typescript/core/types/statement.ts +47 -0
  278. package/typescript/core/types/value.ts +29 -0
  279. package/typescript/index.ts +8 -1
  280. package/typescript/pkg/devalang_core.d.ts +4 -0
  281. package/typescript/pkg/devalang_core.ts +65 -0
  282. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  283. package/typescript/scripts/postinstall.ts +85 -0
  284. package/typescript/scripts/version/bump.ts +0 -1
  285. package/typescript/scripts/version/index.ts +0 -1
  286. package/docs/COMMANDS.md +0 -85
  287. package/docs/CONFIG.md +0 -30
  288. package/docs/SYNTAX.md +0 -210
  289. package/out-tsc/bin/devalang.exe +0 -0
  290. package/out-tsc/scripts/postbuild.js +0 -11
  291. package/rust/cli/build.rs +0 -137
  292. package/rust/cli/check.rs +0 -117
  293. package/rust/cli/play.rs +0 -193
  294. package/rust/config/loader.rs +0 -13
  295. package/rust/core/audio/engine.rs +0 -203
  296. package/rust/core/audio/evaluator.rs +0 -31
  297. package/rust/core/audio/interpreter/arrow_call.rs +0 -129
  298. package/rust/core/audio/interpreter/call.rs +0 -64
  299. package/rust/core/audio/interpreter/let_.rs +0 -19
  300. package/rust/core/audio/interpreter/load.rs +0 -18
  301. package/rust/core/audio/interpreter/loop_.rs +0 -67
  302. package/rust/core/audio/interpreter/sleep.rs +0 -36
  303. package/rust/core/audio/interpreter/spawn.rs +0 -66
  304. package/rust/core/audio/interpreter/tempo.rs +0 -16
  305. package/rust/core/audio/interpreter/trigger.rs +0 -69
  306. package/rust/core/audio/renderer.rs +0 -54
  307. package/rust/core/parser/driver.rs +0 -331
  308. package/rust/core/preprocessor/loader.rs +0 -193
  309. package/rust/core/preprocessor/processor.rs +0 -76
  310. package/rust/core/shared/duration.rs +0 -8
  311. package/rust/core/shared/mod.rs +0 -2
  312. package/rust/core/shared/value.rs +0 -18
  313. package/rust/core/store/export.rs +0 -28
  314. package/rust/core/store/import.rs +0 -28
  315. package/rust/core/store/variable.rs +0 -28
  316. package/rust/core/utils/mod.rs +0 -2
  317. package/rust/core/utils/path.rs +0 -31
  318. package/rust/core/utils/validation.rs +0 -37
  319. package/rust/utils/file.rs +0 -35
  320. package/rust/utils/signature.rs +0 -17
  321. package/rust/utils/version.rs +0 -15
  322. package/typescript/scripts/postbuild.ts +0 -8
@@ -1,25 +1,38 @@
1
- use std::{ collections::HashMap, fs::create_dir_all };
2
- use crate::core::{ debugger::Debugger, preprocessor::module::Module };
3
-
4
- pub fn write_store_log_file(output_dir: &str, file_name: &str, modules: HashMap<String, Module>) {
5
- let debugger = Debugger::new();
6
- let mut content = String::new();
7
-
8
- let log_directory = format!("{}/logs", output_dir);
9
- create_dir_all(&log_directory).expect("Failed to create log directory");
10
-
11
- for (path, module) in modules {
12
- content.push_str(&format!("--- Module: {} ---\n", path));
13
-
14
- for (index, var_value) in module.variable_table.variables.iter().enumerate() {
15
- let var_name = var_value.0.clone();
16
- let var_data = var_value.1;
17
-
18
- content.push_str(&format!("{}: {:?} = {:?}\n", index + 1, var_name, var_data));
19
- }
20
-
21
- content.push_str("\n");
22
- }
23
-
24
- debugger.write_log_file(&log_directory, file_name, &content);
25
- }
1
+ use crate::core::debugger::Debugger;
2
+ use devalang_types::{FunctionTable, VariableTable};
3
+ use std::fs::create_dir_all;
4
+
5
+ pub fn write_variables_log_file(output_dir: &str, file_name: &str, variables: VariableTable) {
6
+ let debugger = Debugger::new();
7
+ let mut content = String::new();
8
+
9
+ let log_directory = format!("{}/logs", output_dir);
10
+ create_dir_all(&log_directory).expect("Failed to create log directory");
11
+
12
+ for (var_name, var_data) in variables.variables {
13
+ content.push_str(&format!("{:?} = {:?}\n", var_name, var_data));
14
+ }
15
+
16
+ content.push('\n');
17
+
18
+ debugger.write_log_file(&log_directory, file_name, &content);
19
+ }
20
+
21
+ pub fn write_function_log_file(output_dir: &str, file_name: &str, functions: FunctionTable) {
22
+ let debugger = Debugger::new();
23
+ let mut content = String::new();
24
+
25
+ let log_directory = format!("{}/logs", output_dir);
26
+ create_dir_all(&log_directory).expect("Failed to create log directory");
27
+
28
+ for (_index, function) in functions.functions {
29
+ content.push_str(&format!(
30
+ "'{}' = [{:?}] => {:?}\n",
31
+ function.name, function.parameters, function.body
32
+ ));
33
+ }
34
+
35
+ content.push('\n');
36
+
37
+ debugger.write_log_file(&log_directory, file_name, &content);
38
+ }
@@ -1,15 +1,50 @@
1
- use crate::core::parser::{ statement::{ Statement, StatementKind }, driver::Parser };
1
+ use crate::core::parser::{
2
+ driver::parser::Parser,
3
+ statement::{Statement, StatementKind},
4
+ };
5
+ use devalang_types::Value;
6
+ use serde::{Deserialize, Serialize};
2
7
 
3
8
  pub struct ErrorHandler {
4
9
  errors: Vec<Error>,
5
10
  }
6
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)]
7
36
  pub struct Error {
8
37
  pub message: String,
9
38
  pub line: usize,
10
39
  pub column: usize,
11
40
  }
12
41
 
42
+ impl Default for ErrorHandler {
43
+ fn default() -> Self {
44
+ Self::new()
45
+ }
46
+ }
47
+
13
48
  impl ErrorHandler {
14
49
  pub fn new() -> Self {
15
50
  Self { errors: Vec::new() }
@@ -36,25 +71,199 @@ impl ErrorHandler {
36
71
  self.errors.clear();
37
72
  }
38
73
 
39
- pub fn detect_from_statements(&mut self, parser: &mut Parser, statements: &[Statement]) {
74
+ pub fn detect_from_statements(&mut self, _parser: &mut Parser, statements: &[Statement]) {
40
75
  for stmt in statements {
41
76
  match &stmt.kind {
42
77
  StatementKind::Unknown => {
43
- self.add_error(
44
- "Unknown statement".to_string(),
45
- stmt.line,
46
- stmt.column
47
- );
78
+ self.add_error("Unknown statement".to_string(), stmt.line, stmt.column);
48
79
  }
49
80
  StatementKind::Error { message } => {
50
- self.add_error(
51
- message.clone(),
52
- stmt.line,
53
- stmt.column
54
- );
81
+ self.add_error(message.clone(), stmt.line, stmt.column);
55
82
  }
56
83
  _ => {}
57
84
  }
58
85
  }
59
86
  }
60
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
+ }
@@ -0,0 +1,59 @@
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
+ }
@@ -1,29 +1,80 @@
1
- use crate::core::lexer::token::{ Token, TokenKind };
1
+ use crate::core::lexer::token::{Token, TokenKind};
2
2
 
3
3
  pub fn handle_arrow_lexer(
4
- char: char,
4
+ ch: char,
5
5
  chars: &mut std::iter::Peekable<std::str::Chars>,
6
6
  current_indent: &mut usize,
7
- indent_stack: &mut Vec<usize>,
7
+ _indent_stack: &mut [usize],
8
8
  tokens: &mut Vec<Token>,
9
9
  line: &mut usize,
10
- column: &mut usize
10
+ column: &mut usize,
11
11
  ) {
12
- let mut arrow_call = char.to_string();
13
-
14
- while let Some(&c) = chars.peek() {
12
+ // If next char is '>', this is an arrow '->'.
13
+ if let Some(&c) = chars.peek() {
15
14
  if c == '>' {
15
+ let mut arrow_call = ch.to_string();
16
16
  chars.next();
17
17
  arrow_call.push(c);
18
18
  *column += 1;
19
- } else {
20
- break;
19
+
20
+ tokens.push(Token {
21
+ kind: TokenKind::Arrow,
22
+ lexeme: arrow_call,
23
+ line: *line,
24
+ column: *column,
25
+ indent: *current_indent,
26
+ });
27
+ return;
28
+ }
29
+ }
30
+
31
+ // Otherwise, treat '-' as the start of a negative number if followed by digits.
32
+ let mut lexeme = String::from("-");
33
+ if let Some(&next) = chars.peek() {
34
+ if next.is_ascii_digit() {
35
+ // consume digits
36
+ while let Some(&d) = chars.peek() {
37
+ if d.is_ascii_digit() {
38
+ chars.next();
39
+ lexeme.push(d);
40
+ *column += 1;
41
+ } else {
42
+ break;
43
+ }
44
+ }
45
+ // optional decimal part
46
+ if let Some(&dot) = chars.peek() {
47
+ if dot == '.' {
48
+ chars.next();
49
+ lexeme.push(dot);
50
+ *column += 1;
51
+ while let Some(&d) = chars.peek() {
52
+ if d.is_ascii_digit() {
53
+ chars.next();
54
+ lexeme.push(d);
55
+ *column += 1;
56
+ } else {
57
+ break;
58
+ }
59
+ }
60
+ }
61
+ }
62
+
63
+ tokens.push(Token {
64
+ kind: TokenKind::Number,
65
+ lexeme,
66
+ line: *line,
67
+ column: *column,
68
+ indent: *current_indent,
69
+ });
70
+ return;
21
71
  }
22
72
  }
23
73
 
74
+ // Fallback: lone '-' not part of '->' or a number; emit Unknown to avoid mis-parsing as Arrow
24
75
  tokens.push(Token {
25
- kind: TokenKind::Arrow,
26
- lexeme: arrow_call,
76
+ kind: TokenKind::Unknown,
77
+ lexeme: "-".to_string(),
27
78
  line: *line,
28
79
  column: *column,
29
80
  indent: *current_indent,
@@ -1,17 +1,17 @@
1
1
  use crate::core::lexer::token::{Token, TokenKind};
2
2
 
3
3
  pub fn handle_at_lexer(
4
- char: char,
5
- chars: &mut std::iter::Peekable<std::str::Chars>,
4
+ ch: char,
5
+ _chars: &mut std::iter::Peekable<std::str::Chars>,
6
6
  current_indent: &mut usize,
7
- indent_stack: &mut Vec<usize>,
7
+ _indent_stack: &mut [usize],
8
8
  tokens: &mut Vec<Token>,
9
9
  line: &mut usize,
10
- column: &mut usize
10
+ column: &mut usize,
11
11
  ) {
12
12
  tokens.push(Token {
13
13
  kind: TokenKind::At,
14
- lexeme: char.to_string(),
14
+ lexeme: ch.to_string(),
15
15
  line: *line,
16
16
  column: *column,
17
17
  indent: *current_indent,
@@ -1,17 +1,17 @@
1
- use crate::core::lexer::token::{ Token, TokenKind };
1
+ use crate::core::lexer::token::{Token, TokenKind};
2
2
 
3
3
  pub fn handle_rbrace_lexer(
4
- char: char,
5
- chars: &mut std::iter::Peekable<std::str::Chars>,
4
+ ch: char,
5
+ _chars: &mut std::iter::Peekable<std::str::Chars>,
6
6
  current_indent: &mut usize,
7
- indent_stack: &mut Vec<usize>,
7
+ _indent_stack: &mut [usize],
8
8
  tokens: &mut Vec<Token>,
9
9
  line: &mut usize,
10
- column: &mut usize
10
+ column: &mut usize,
11
11
  ) {
12
12
  tokens.push(Token {
13
13
  kind: TokenKind::RBrace,
14
- lexeme: char.to_string(),
14
+ lexeme: ch.to_string(),
15
15
  line: *line,
16
16
  column: *column,
17
17
  indent: *current_indent,
@@ -21,17 +21,17 @@ pub fn handle_rbrace_lexer(
21
21
  }
22
22
 
23
23
  pub fn handle_lbrace_lexer(
24
- char: char,
25
- chars: &mut std::iter::Peekable<std::str::Chars>,
24
+ ch: char,
25
+ _chars: &mut std::iter::Peekable<std::str::Chars>,
26
26
  current_indent: &mut usize,
27
- indent_stack: &mut Vec<usize>,
27
+ _indent_stack: &mut [usize],
28
28
  tokens: &mut Vec<Token>,
29
29
  line: &mut usize,
30
- column: &mut usize
30
+ column: &mut usize,
31
31
  ) {
32
32
  tokens.push(Token {
33
33
  kind: TokenKind::LBrace,
34
- lexeme: char.to_string(),
34
+ lexeme: ch.to_string(),
35
35
  line: *line,
36
36
  column: *column,
37
37
  indent: *current_indent,
@@ -1,17 +1,17 @@
1
1
  use crate::core::lexer::token::{Token, TokenKind};
2
2
 
3
3
  pub fn handle_colon_lexer(
4
- char: char,
5
- chars: &mut std::iter::Peekable<std::str::Chars>,
4
+ ch: char,
5
+ _chars: &mut std::iter::Peekable<std::str::Chars>,
6
6
  current_indent: &mut usize,
7
- indent_stack: &mut Vec<usize>,
7
+ _indent_stack: &mut [usize],
8
8
  tokens: &mut Vec<Token>,
9
9
  line: &mut usize,
10
- column: &mut usize
10
+ column: &mut usize,
11
11
  ) {
12
12
  tokens.push(Token {
13
13
  kind: TokenKind::Colon,
14
- lexeme: char.to_string(),
14
+ lexeme: ch.to_string(),
15
15
  line: *line,
16
16
  column: *column,
17
17
  indent: *current_indent,
@@ -1,13 +1,13 @@
1
1
  use crate::core::lexer::token::{Token, TokenKind};
2
2
 
3
3
  pub fn handle_comment_lexer(
4
- char: char,
4
+ _char: char,
5
5
  chars: &mut std::iter::Peekable<std::str::Chars>,
6
6
  current_indent: &mut usize,
7
- indent_stack: &mut Vec<usize>,
7
+ _indent_stack: &mut [usize],
8
8
  tokens: &mut Vec<Token>,
9
9
  line: &mut usize,
10
- column: &mut usize
10
+ column: &mut usize,
11
11
  ) {
12
12
  let mut comment = String::new();
13
13
 
@@ -1,17 +1,17 @@
1
- use crate::core::lexer::token::{ Token, TokenKind };
1
+ use crate::core::lexer::token::{Token, TokenKind};
2
2
 
3
3
  pub fn handle_dot_lexer(
4
- char: char,
5
- chars: &mut std::iter::Peekable<std::str::Chars>,
4
+ ch: char,
5
+ _chars: &mut std::iter::Peekable<std::str::Chars>,
6
6
  current_indent: &mut usize,
7
- indent_stack: &mut Vec<usize>,
7
+ _indent_stack: &mut [usize],
8
8
  tokens: &mut Vec<Token>,
9
9
  line: &mut usize,
10
- column: &mut usize
10
+ column: &mut usize,
11
11
  ) {
12
12
  tokens.push(Token {
13
13
  kind: TokenKind::Dot,
14
- lexeme: char.to_string(),
14
+ lexeme: ch.to_string(),
15
15
  line: *line,
16
16
  column: *column,
17
17
  indent: *current_indent,