@devaloop/devalang 0.0.1-alpha.15 → 0.0.1-alpha.16-hotfix.0

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 (173) hide show
  1. package/.devalang +2 -0
  2. package/.github/workflows/ci.yml +92 -0
  3. package/Cargo.toml +60 -58
  4. package/README.md +1 -1
  5. package/docs/CHANGELOG.md +34 -1
  6. package/docs/CONTRIBUTING.md +101 -1
  7. package/docs/ROADMAP.md +1 -1
  8. package/docs/TODO.md +1 -1
  9. package/examples/automation.deva +1 -3
  10. package/examples/bank.deva +4 -4
  11. package/examples/events.deva +12 -0
  12. package/examples/function.deva +4 -4
  13. package/examples/index.deva +3 -5
  14. package/examples/loop.deva +5 -11
  15. package/examples/pattern.deva +8 -0
  16. package/examples/plugin.deva +12 -11
  17. package/examples/variables.deva +1 -1
  18. package/out-tsc/bin/index.js +51 -7
  19. package/out-tsc/index.js +3 -1
  20. package/out-tsc/scripts/postbuild.js +9 -10
  21. package/out-tsc/scripts/postinstall.js +49 -0
  22. package/package.json +12 -4
  23. package/project-version.json +3 -3
  24. package/rust/cli/bank.rs +462 -455
  25. package/rust/cli/build.rs +252 -199
  26. package/rust/cli/check.rs +221 -180
  27. package/rust/cli/driver.rs +297 -292
  28. package/rust/cli/generator.rs +1 -0
  29. package/rust/cli/init.rs +87 -79
  30. package/rust/cli/install.rs +35 -32
  31. package/rust/cli/login.rs +127 -134
  32. package/rust/cli/mod.rs +13 -11
  33. package/rust/cli/play.rs +1123 -218
  34. package/rust/cli/telemetry.rs +19 -0
  35. package/rust/cli/template.rs +69 -57
  36. package/rust/cli/update.rs +6 -4
  37. package/rust/common/api.rs +5 -5
  38. package/rust/common/mod.rs +3 -3
  39. package/rust/config/driver.rs +118 -94
  40. package/rust/config/loader.rs +165 -156
  41. package/rust/config/mod.rs +4 -2
  42. package/rust/config/settings.rs +91 -0
  43. package/rust/config/stats.rs +257 -0
  44. package/rust/core/audio/engine.rs +696 -659
  45. package/rust/core/audio/evaluator.rs +263 -132
  46. package/rust/core/audio/interpreter/arrow_call.rs +198 -187
  47. package/rust/core/audio/interpreter/call.rs +98 -95
  48. package/rust/core/audio/interpreter/condition.rs +70 -71
  49. package/rust/core/audio/interpreter/driver.rs +487 -231
  50. package/rust/core/audio/interpreter/function.rs +26 -21
  51. package/rust/core/audio/interpreter/let_.rs +38 -26
  52. package/rust/core/audio/interpreter/load.rs +18 -18
  53. package/rust/core/audio/interpreter/loop_.rs +113 -106
  54. package/rust/core/audio/interpreter/mod.rs +14 -14
  55. package/rust/core/audio/interpreter/sleep.rs +27 -28
  56. package/rust/core/audio/interpreter/spawn.rs +105 -102
  57. package/rust/core/audio/interpreter/tempo.rs +19 -16
  58. package/rust/core/audio/interpreter/trigger.rs +239 -210
  59. package/rust/core/audio/loader/mod.rs +1 -1
  60. package/rust/core/audio/loader/trigger.rs +100 -94
  61. package/rust/core/audio/mod.rs +7 -7
  62. package/rust/core/audio/player.rs +64 -64
  63. package/rust/core/audio/renderer.rs +56 -53
  64. package/rust/core/audio/special/easing.rs +189 -120
  65. package/rust/core/audio/special/env.rs +43 -41
  66. package/rust/core/audio/special/math.rs +102 -92
  67. package/rust/core/audio/special/mod.rs +9 -9
  68. package/rust/core/audio/special/modulator.rs +143 -120
  69. package/rust/core/builder/mod.rs +80 -85
  70. package/rust/core/debugger/lexer.rs +27 -27
  71. package/rust/core/debugger/mod.rs +24 -23
  72. package/rust/core/debugger/module.rs +55 -47
  73. package/rust/core/debugger/preprocessor.rs +27 -27
  74. package/rust/core/debugger/store.rs +40 -39
  75. package/rust/core/error/mod.rs +80 -69
  76. package/rust/core/lexer/handler/arrow.rs +82 -82
  77. package/rust/core/lexer/handler/at.rs +21 -21
  78. package/rust/core/lexer/handler/brace.rs +41 -41
  79. package/rust/core/lexer/handler/colon.rs +21 -21
  80. package/rust/core/lexer/handler/comment.rs +30 -30
  81. package/rust/core/lexer/handler/dot.rs +21 -21
  82. package/rust/core/lexer/handler/driver.rs +337 -292
  83. package/rust/core/lexer/handler/identifier.rs +46 -43
  84. package/rust/core/lexer/handler/indent.rs +66 -66
  85. package/rust/core/lexer/handler/mod.rs +16 -16
  86. package/rust/core/lexer/handler/newline.rs +23 -23
  87. package/rust/core/lexer/handler/number.rs +31 -31
  88. package/rust/core/lexer/handler/operator.rs +46 -46
  89. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  90. package/rust/core/lexer/handler/slash.rs +21 -21
  91. package/rust/core/lexer/handler/string.rs +63 -63
  92. package/rust/core/lexer/mod.rs +54 -51
  93. package/rust/core/lexer/token.rs +97 -94
  94. package/rust/core/mod.rs +11 -11
  95. package/rust/core/parser/driver.rs +513 -490
  96. package/rust/core/parser/handler/arrow_call.rs +233 -227
  97. package/rust/core/parser/handler/at.rs +245 -162
  98. package/rust/core/parser/handler/bank.rs +94 -69
  99. package/rust/core/parser/handler/condition.rs +80 -74
  100. package/rust/core/parser/handler/dot.rs +143 -135
  101. package/rust/core/parser/handler/identifier/automate.rs +257 -194
  102. package/rust/core/parser/handler/identifier/call.rs +91 -88
  103. package/rust/core/parser/handler/identifier/emit.rs +66 -0
  104. package/rust/core/parser/handler/identifier/function.rs +100 -91
  105. package/rust/core/parser/handler/identifier/group.rs +85 -75
  106. package/rust/core/parser/handler/identifier/let_.rs +158 -143
  107. package/rust/core/parser/handler/identifier/mod.rs +54 -56
  108. package/rust/core/parser/handler/identifier/on.rs +98 -0
  109. package/rust/core/parser/handler/identifier/print.rs +52 -29
  110. package/rust/core/parser/handler/identifier/sleep.rs +36 -33
  111. package/rust/core/parser/handler/identifier/spawn.rs +91 -88
  112. package/rust/core/parser/handler/identifier/synth.rs +65 -63
  113. package/rust/core/parser/handler/loop_.rs +170 -89
  114. package/rust/core/parser/handler/mod.rs +8 -8
  115. package/rust/core/parser/handler/tempo.rs +53 -47
  116. package/rust/core/parser/mod.rs +4 -4
  117. package/rust/core/parser/statement.rs +142 -113
  118. package/rust/core/plugin/loader.rs +123 -48
  119. package/rust/core/plugin/mod.rs +2 -1
  120. package/rust/core/plugin/runner.rs +296 -0
  121. package/rust/core/preprocessor/loader.rs +515 -326
  122. package/rust/core/preprocessor/mod.rs +4 -4
  123. package/rust/core/preprocessor/module.rs +60 -58
  124. package/rust/core/preprocessor/processor.rs +99 -101
  125. package/rust/core/preprocessor/resolver/bank.rs +51 -48
  126. package/rust/core/preprocessor/resolver/call.rs +100 -101
  127. package/rust/core/preprocessor/resolver/condition.rs +97 -97
  128. package/rust/core/preprocessor/resolver/driver.rs +310 -280
  129. package/rust/core/preprocessor/resolver/function.rs +69 -68
  130. package/rust/core/preprocessor/resolver/group.rs +96 -91
  131. package/rust/core/preprocessor/resolver/let_.rs +32 -28
  132. package/rust/core/preprocessor/resolver/loop_.rs +320 -121
  133. package/rust/core/preprocessor/resolver/mod.rs +15 -15
  134. package/rust/core/preprocessor/resolver/spawn.rs +76 -73
  135. package/rust/core/preprocessor/resolver/synth.rs +56 -50
  136. package/rust/core/preprocessor/resolver/tempo.rs +50 -49
  137. package/rust/core/preprocessor/resolver/trigger.rs +113 -115
  138. package/rust/core/preprocessor/resolver/value.rs +81 -81
  139. package/rust/core/shared/duration.rs +9 -9
  140. package/rust/core/shared/mod.rs +3 -3
  141. package/rust/core/shared/value.rs +35 -32
  142. package/rust/core/store/function.rs +34 -34
  143. package/rust/core/store/global.rs +55 -38
  144. package/rust/core/store/mod.rs +5 -5
  145. package/rust/core/store/variable.rs +37 -34
  146. package/rust/core/utils/mod.rs +2 -2
  147. package/rust/core/utils/path.rs +37 -31
  148. package/rust/core/utils/validation.rs +35 -36
  149. package/rust/installer/addon.rs +84 -80
  150. package/rust/installer/bank.rs +62 -65
  151. package/rust/installer/mod.rs +5 -5
  152. package/rust/installer/plugin.rs +54 -55
  153. package/rust/installer/utils.rs +56 -56
  154. package/rust/lib.rs +156 -164
  155. package/rust/main.rs +250 -144
  156. package/rust/utils/error.rs +200 -51
  157. package/rust/utils/file.rs +38 -35
  158. package/rust/utils/first_usage.rs +76 -0
  159. package/rust/utils/logger.rs +195 -143
  160. package/rust/utils/mod.rs +9 -7
  161. package/rust/utils/signature.rs +19 -17
  162. package/rust/utils/spinner.rs +22 -19
  163. package/rust/utils/telemetry.rs +292 -0
  164. package/rust/utils/watcher.rs +34 -33
  165. package/templates/minimal/README.md +97 -121
  166. package/templates/welcome/README.md +97 -121
  167. package/typescript/bin/index.ts +19 -5
  168. package/typescript/index.ts +3 -1
  169. package/typescript/scripts/postbuild.ts +10 -6
  170. package/typescript/scripts/postinstall.ts +56 -0
  171. package/typescript/scripts/version/bump.ts +0 -1
  172. package/typescript/scripts/version/index.ts +0 -1
  173. package/out-tsc/bin/devalang.exe +0 -0
package/rust/cli/check.rs CHANGED
@@ -1,180 +1,221 @@
1
- use crate::{
2
- config::driver::Config,
3
- core::{
4
- debugger::{
5
- lexer::write_lexer_log_file,
6
- module::{ write_module_function_log_file, write_module_variable_log_file },
7
- preprocessor::write_preprocessor_log_file,
8
- store::{ write_function_log_file, write_variables_log_file },
9
- },
10
- preprocessor::loader::ModuleLoader,
11
- store::global::GlobalStore,
12
- utils::path::{ find_entry_file, normalize_path },
13
- },
14
- utils::{
15
- error::collect_errors_recursively,
16
- logger::{ LogLevel, Logger },
17
- spinner::with_spinner,
18
- watcher::watch_directory,
19
- },
20
- };
21
- use std::{ thread, time::Duration };
22
-
23
- #[cfg(feature = "cli")]
24
- pub fn handle_check_command(
25
- config: Option<Config>,
26
- entry: Option<String>,
27
- output: Option<String>,
28
- watch: bool,
29
- debug: bool
30
- ) {
31
- let fetched_entry = if entry.is_none() {
32
- config
33
- .as_ref()
34
- .and_then(|c| c.defaults.entry.clone())
35
- .unwrap_or_else(|| "".to_string())
36
- } else {
37
- entry.clone().unwrap_or_else(|| "".to_string())
38
- };
39
-
40
- let fetched_output = if output.is_none() {
41
- config
42
- .as_ref()
43
- .and_then(|c| c.defaults.output.clone())
44
- .unwrap_or_else(|| "".to_string())
45
- } else {
46
- output.clone().unwrap_or_else(|| "".to_string())
47
- };
48
-
49
- let fetched_watch = if watch {
50
- watch
51
- } else {
52
- config
53
- .as_ref()
54
- .and_then(|c| c.defaults.watch)
55
- .unwrap_or(false)
56
- };
57
-
58
- let logger = Logger::new();
59
-
60
- if fetched_entry.is_empty() {
61
- logger.log_message(
62
- LogLevel::Error,
63
- "Entry path is not specified. Please provide a valid entry path."
64
- );
65
- std::process::exit(1);
66
- }
67
- if fetched_output.is_empty() {
68
- logger.log_message(
69
- LogLevel::Error,
70
- "Output directory is not specified. Please provide a valid output directory."
71
- );
72
- std::process::exit(1);
73
- }
74
-
75
- let entry_file = find_entry_file(&fetched_entry).unwrap_or_else(|| {
76
- logger.log_message(
77
- LogLevel::Error,
78
- &format!("❌ index.deva not found in directory: {}", fetched_entry)
79
- );
80
- std::process::exit(1);
81
- });
82
-
83
- // SECTION Begin check
84
- if fetched_watch {
85
- begin_check(entry_file.clone(), fetched_output.clone(), debug);
86
-
87
- logger.log_message(
88
- LogLevel::Watcher,
89
- &format!("Watching for changes in '{}'...", fetched_entry)
90
- );
91
-
92
- watch_directory(entry_file.clone(), move || {
93
- logger.log_message(LogLevel::Watcher, "Detected changes, re-checking...");
94
-
95
- begin_check(entry_file.clone(), fetched_output.clone(), debug);
96
- }).unwrap();
97
- } else {
98
- begin_check(entry_file.clone(), fetched_output.clone(), debug);
99
- }
100
- }
101
-
102
- fn begin_check(entry: String, output: String, debug: bool) {
103
- let spinner = with_spinner("Checking...", || {
104
- thread::sleep(Duration::from_millis(800));
105
- });
106
-
107
- let duration = std::time::Instant::now();
108
-
109
- let normalized_entry_file = normalize_path(&entry);
110
- let normalized_output_dir = normalize_path(&output);
111
-
112
- let mut global_store = GlobalStore::new();
113
- let module_loader = ModuleLoader::new(&normalized_entry_file, &normalized_output_dir);
114
-
115
- // SECTION Load
116
- // NOTE: We don't use modules in the check command, but we still need to load them
117
- let modules = module_loader.load_all_modules(&mut global_store);
118
-
119
- // Debugging: Log loaded modules and errors
120
- let logger = Logger::new();
121
- logger.log_message(LogLevel::Info, "Loaded modules:");
122
- for (module_name, _) in &modules.0 {
123
- logger.log_message(LogLevel::Info, &format!("- {}", module_name));
124
- }
125
-
126
- if debug {
127
- for (module_path, module) in global_store.modules.clone() {
128
- write_module_variable_log_file(
129
- &normalized_output_dir,
130
- &module_path,
131
- &module.variable_table
132
- );
133
- write_module_function_log_file(
134
- &normalized_output_dir,
135
- &module_path,
136
- &module.function_table
137
- );
138
- }
139
-
140
- write_lexer_log_file(&normalized_output_dir, "lexer_tokens.log", modules.0.clone());
141
- write_preprocessor_log_file(
142
- &normalized_output_dir,
143
- "resolved_statements.log",
144
- modules.1.clone()
145
- );
146
- write_variables_log_file(
147
- &normalized_output_dir,
148
- "global_variables.log",
149
- global_store.variables.clone()
150
- );
151
- write_function_log_file(
152
- &normalized_output_dir,
153
- "global_functions.log",
154
- global_store.functions.clone()
155
- );
156
- }
157
-
158
- let mut all_errors = Vec::new();
159
- for (_, statements) in &modules.1 {
160
- all_errors.extend(collect_errors_recursively(statements));
161
- }
162
-
163
- if !all_errors.is_empty() {
164
- logger.log_message(LogLevel::Error, "Errors detected during check:");
165
- for error in all_errors {
166
- logger.log_message(LogLevel::Error, &format!("- {}", error.message));
167
- }
168
- } else {
169
- logger.log_message(LogLevel::Success, "No errors detected.");
170
- }
171
-
172
- let success_message = format!(
173
- "Check completed successfully in {:.2?}. Output files written to: '{}'",
174
- duration.elapsed(),
175
- normalized_output_dir
176
- );
177
-
178
- spinner.finish_and_clear();
179
- logger.log_message(LogLevel::Success, &success_message);
180
- }
1
+ use crate::{
2
+ config::driver::ProjectConfig,
3
+ core::{
4
+ debugger::{
5
+ lexer::write_lexer_log_file,
6
+ module::{write_module_function_log_file, write_module_variable_log_file},
7
+ preprocessor::write_preprocessor_log_file,
8
+ store::{write_function_log_file, write_variables_log_file},
9
+ },
10
+ preprocessor::loader::ModuleLoader,
11
+ store::global::GlobalStore,
12
+ utils::path::{find_entry_file, normalize_path},
13
+ },
14
+ utils::{
15
+ logger::{LogLevel, Logger},
16
+ spinner::with_spinner,
17
+ watcher::watch_directory,
18
+ },
19
+ };
20
+ use std::{thread, time::Duration};
21
+
22
+ #[cfg(feature = "cli")]
23
+ pub fn handle_check_command(
24
+ config: Option<ProjectConfig>,
25
+ entry: Option<String>,
26
+ output: Option<String>,
27
+ watch: bool,
28
+ debug: bool,
29
+ ) -> Result<(), String> {
30
+ let fetched_entry = if entry.is_none() {
31
+ config
32
+ .as_ref()
33
+ .and_then(|c| c.defaults.entry.clone())
34
+ .unwrap_or_else(|| "".to_string())
35
+ } else {
36
+ entry.clone().unwrap_or_else(|| "".to_string())
37
+ };
38
+
39
+ let fetched_output = if output.is_none() {
40
+ config
41
+ .as_ref()
42
+ .and_then(|c| c.defaults.output.clone())
43
+ .unwrap_or_else(|| "".to_string())
44
+ } else {
45
+ output.clone().unwrap_or_else(|| "".to_string())
46
+ };
47
+
48
+ let fetched_watch = if watch {
49
+ watch
50
+ } else {
51
+ config
52
+ .as_ref()
53
+ .and_then(|c| c.defaults.watch)
54
+ .unwrap_or(false)
55
+ };
56
+
57
+ let logger = Logger::new();
58
+
59
+ if fetched_entry.is_empty() {
60
+ logger.log_message(
61
+ LogLevel::Error,
62
+ "Entry path is not specified. Please provide a valid entry path.",
63
+ );
64
+ return Err("missing entry path".to_string());
65
+ }
66
+ if fetched_output.is_empty() {
67
+ logger.log_message(
68
+ LogLevel::Error,
69
+ "Output directory is not specified. Please provide a valid output directory.",
70
+ );
71
+ return Err("missing output directory".to_string());
72
+ }
73
+
74
+ let entry_file = match find_entry_file(&fetched_entry) {
75
+ Some(p) => p,
76
+ None => {
77
+ logger.log_message(
78
+ LogLevel::Error,
79
+ &format!("❌ index.deva not found in directory: {}", fetched_entry),
80
+ );
81
+ return Err("index.deva not found".to_string());
82
+ }
83
+ };
84
+
85
+ // SECTION Begin check
86
+ if fetched_watch {
87
+ let _ = begin_check(
88
+ entry_file.clone(),
89
+ fetched_output.clone(),
90
+ debug,
91
+ config.clone(),
92
+ );
93
+
94
+ logger.log_message(
95
+ LogLevel::Watcher,
96
+ &format!("Watching for changes in '{}'...", fetched_entry),
97
+ );
98
+
99
+ let cfg_for_watch = config.clone();
100
+ watch_directory(entry_file.clone(), move || {
101
+ logger.log_message(LogLevel::Watcher, "Detected changes, re-checking...");
102
+ if let Err(e) = begin_check(
103
+ entry_file.clone(),
104
+ fetched_output.clone(),
105
+ debug,
106
+ cfg_for_watch.clone(),
107
+ ) {
108
+ eprintln!("[check] failed: {}", e);
109
+ }
110
+ })
111
+ .unwrap();
112
+ } else {
113
+ begin_check(
114
+ entry_file.clone(),
115
+ fetched_output.clone(),
116
+ debug,
117
+ config.clone(),
118
+ )?;
119
+ }
120
+ Ok(())
121
+ }
122
+
123
+ fn begin_check(
124
+ entry: String,
125
+ output: String,
126
+ debug: bool,
127
+ config: Option<ProjectConfig>,
128
+ ) -> Result<(), String> {
129
+ let spinner = with_spinner("Checking...", || {
130
+ thread::sleep(Duration::from_millis(800));
131
+ });
132
+
133
+ let duration = std::time::Instant::now();
134
+
135
+ let normalized_entry_file = normalize_path(&entry);
136
+ let normalized_output_dir = normalize_path(&output);
137
+
138
+ let mut global_store = GlobalStore::new();
139
+ let module_loader = ModuleLoader::new(&normalized_entry_file, &normalized_output_dir);
140
+
141
+ // SECTION Load
142
+ // NOTE: We don't use modules in the check command, but we still need to load them
143
+ let modules = module_loader.load_all_modules(&mut global_store);
144
+
145
+ // Debugging: Log loaded modules and errors
146
+ let logger = Logger::new();
147
+ logger.log_message(LogLevel::Info, "Loaded modules:");
148
+ for (module_name, _) in &modules.0 {
149
+ logger.log_message(LogLevel::Info, &format!("- {}", module_name));
150
+ }
151
+
152
+ if debug {
153
+ for (module_path, module) in global_store.modules.clone() {
154
+ write_module_variable_log_file(
155
+ &normalized_output_dir,
156
+ &module_path,
157
+ &module.variable_table,
158
+ );
159
+ write_module_function_log_file(
160
+ &normalized_output_dir,
161
+ &module_path,
162
+ &module.function_table,
163
+ );
164
+ }
165
+
166
+ write_lexer_log_file(
167
+ &normalized_output_dir,
168
+ "lexer_tokens.log",
169
+ modules.0.clone(),
170
+ );
171
+ write_preprocessor_log_file(
172
+ &normalized_output_dir,
173
+ "resolved_statements.log",
174
+ modules.1.clone(),
175
+ );
176
+ write_variables_log_file(
177
+ &normalized_output_dir,
178
+ "global_variables.log",
179
+ global_store.variables.clone(),
180
+ );
181
+ write_function_log_file(
182
+ &normalized_output_dir,
183
+ "global_functions.log",
184
+ global_store.functions.clone(),
185
+ );
186
+ }
187
+
188
+ let all_errors = crate::utils::error::collect_all_errors_with_modules(&modules.1);
189
+
190
+ let (warnings, criticals) = crate::utils::error::partition_errors(all_errors);
191
+ crate::utils::error::log_errors_with_stack("Check", &warnings, &criticals);
192
+
193
+ if !criticals.is_empty() {
194
+ spinner.finish_and_clear();
195
+ return Err("check failed with critical errors".to_string());
196
+ } else {
197
+ logger.log_message(LogLevel::Success, "No errors detected.");
198
+
199
+ // Compute and persist rich stats
200
+ let stats = crate::config::stats::compute_from(
201
+ &modules.1,
202
+ &global_store,
203
+ &config,
204
+ Some(&normalized_output_dir),
205
+ );
206
+ crate::config::stats::set_memory_stats(stats.clone());
207
+ if let Err(e) = crate::config::stats::save_to_file(&stats) {
208
+ eprintln!("[stats] failed to save: {}", e);
209
+ }
210
+
211
+ let success_message = format!(
212
+ "Check completed successfully in {:.2?}. Output files written to: '{}'",
213
+ duration.elapsed(),
214
+ normalized_output_dir
215
+ );
216
+
217
+ spinner.finish_and_clear();
218
+ logger.log_message(LogLevel::Success, &success_message);
219
+ Ok(())
220
+ }
221
+ }