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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (177) hide show
  1. package/.devalang +10 -8
  2. package/.github/workflows/ci.yml +92 -0
  3. package/Cargo.toml +60 -58
  4. package/README.md +32 -15
  5. package/docs/CHANGELOG.md +93 -1
  6. package/docs/CONTRIBUTING.md +101 -1
  7. package/docs/ROADMAP.md +2 -2
  8. package/docs/TODO.md +1 -1
  9. package/examples/automation.deva +42 -0
  10. package/examples/bank.deva +4 -4
  11. package/examples/events.deva +12 -0
  12. package/examples/function.deva +4 -4
  13. package/examples/index.deva +39 -25
  14. package/examples/loop.deva +5 -11
  15. package/examples/pattern.deva +8 -0
  16. package/examples/plugin.deva +16 -0
  17. package/examples/variables.deva +1 -1
  18. package/out-tsc/bin/index.js +51 -7
  19. package/out-tsc/index.js +3 -1
  20. package/out-tsc/scripts/postbuild.js +9 -10
  21. package/out-tsc/scripts/postinstall.js +49 -0
  22. package/package.json +12 -4
  23. package/project-version.json +3 -3
  24. package/rust/cli/bank.rs +462 -456
  25. package/rust/cli/build.rs +252 -199
  26. package/rust/cli/check.rs +221 -180
  27. package/rust/cli/driver.rs +297 -292
  28. package/rust/cli/generator.rs +1 -0
  29. package/rust/cli/init.rs +87 -79
  30. package/rust/cli/install.rs +35 -32
  31. package/rust/cli/login.rs +127 -134
  32. package/rust/cli/mod.rs +13 -11
  33. package/rust/cli/play.rs +1123 -218
  34. package/rust/cli/telemetry.rs +19 -0
  35. package/rust/cli/template.rs +69 -57
  36. package/rust/cli/update.rs +6 -4
  37. package/rust/common/api.rs +5 -8
  38. package/rust/common/cdn.rs +3 -6
  39. package/rust/common/mod.rs +3 -3
  40. package/rust/common/sso.rs +3 -6
  41. package/rust/config/driver.rs +118 -94
  42. package/rust/config/loader.rs +165 -156
  43. package/rust/config/mod.rs +4 -2
  44. package/rust/config/settings.rs +91 -0
  45. package/rust/config/stats.rs +257 -0
  46. package/rust/core/audio/engine.rs +696 -518
  47. package/rust/core/audio/evaluator.rs +263 -31
  48. package/rust/core/audio/interpreter/arrow_call.rs +198 -161
  49. package/rust/core/audio/interpreter/automate.rs +18 -0
  50. package/rust/core/audio/interpreter/call.rs +98 -95
  51. package/rust/core/audio/interpreter/condition.rs +70 -71
  52. package/rust/core/audio/interpreter/driver.rs +487 -198
  53. package/rust/core/audio/interpreter/function.rs +26 -21
  54. package/rust/core/audio/interpreter/let_.rs +38 -19
  55. package/rust/core/audio/interpreter/load.rs +18 -18
  56. package/rust/core/audio/interpreter/loop_.rs +113 -73
  57. package/rust/core/audio/interpreter/mod.rs +14 -13
  58. package/rust/core/audio/interpreter/sleep.rs +27 -30
  59. package/rust/core/audio/interpreter/spawn.rs +105 -102
  60. package/rust/core/audio/interpreter/tempo.rs +19 -16
  61. package/rust/core/audio/interpreter/trigger.rs +239 -210
  62. package/rust/core/audio/loader/mod.rs +1 -1
  63. package/rust/core/audio/loader/trigger.rs +100 -97
  64. package/rust/core/audio/mod.rs +7 -6
  65. package/rust/core/audio/player.rs +64 -64
  66. package/rust/core/audio/renderer.rs +56 -53
  67. package/rust/core/audio/special/easing.rs +189 -0
  68. package/rust/core/audio/special/env.rs +43 -0
  69. package/rust/core/audio/special/math.rs +102 -0
  70. package/rust/core/audio/special/mod.rs +9 -0
  71. package/rust/core/audio/special/modulator.rs +143 -0
  72. package/rust/core/builder/mod.rs +80 -85
  73. package/rust/core/debugger/lexer.rs +27 -27
  74. package/rust/core/debugger/mod.rs +24 -23
  75. package/rust/core/debugger/module.rs +55 -47
  76. package/rust/core/debugger/preprocessor.rs +27 -27
  77. package/rust/core/debugger/store.rs +40 -39
  78. package/rust/core/error/mod.rs +80 -66
  79. package/rust/core/lexer/handler/arrow.rs +82 -31
  80. package/rust/core/lexer/handler/at.rs +21 -21
  81. package/rust/core/lexer/handler/brace.rs +41 -41
  82. package/rust/core/lexer/handler/colon.rs +21 -21
  83. package/rust/core/lexer/handler/comment.rs +30 -30
  84. package/rust/core/lexer/handler/dot.rs +21 -21
  85. package/rust/core/lexer/handler/driver.rs +337 -263
  86. package/rust/core/lexer/handler/identifier.rs +46 -42
  87. package/rust/core/lexer/handler/indent.rs +66 -66
  88. package/rust/core/lexer/handler/mod.rs +16 -16
  89. package/rust/core/lexer/handler/newline.rs +23 -23
  90. package/rust/core/lexer/handler/number.rs +31 -31
  91. package/rust/core/lexer/handler/operator.rs +46 -44
  92. package/rust/core/lexer/handler/parenthesis.rs +41 -41
  93. package/rust/core/lexer/handler/slash.rs +21 -21
  94. package/rust/core/lexer/handler/string.rs +63 -63
  95. package/rust/core/lexer/mod.rs +54 -51
  96. package/rust/core/lexer/token.rs +97 -91
  97. package/rust/core/mod.rs +11 -11
  98. package/rust/core/parser/driver.rs +513 -408
  99. package/rust/core/parser/handler/arrow_call.rs +233 -211
  100. package/rust/core/parser/handler/at.rs +245 -162
  101. package/rust/core/parser/handler/bank.rs +94 -69
  102. package/rust/core/parser/handler/condition.rs +80 -74
  103. package/rust/core/parser/handler/dot.rs +143 -135
  104. package/rust/core/parser/handler/identifier/automate.rs +257 -0
  105. package/rust/core/parser/handler/identifier/call.rs +91 -88
  106. package/rust/core/parser/handler/identifier/emit.rs +66 -0
  107. package/rust/core/parser/handler/identifier/function.rs +100 -92
  108. package/rust/core/parser/handler/identifier/group.rs +85 -75
  109. package/rust/core/parser/handler/identifier/let_.rs +158 -127
  110. package/rust/core/parser/handler/identifier/mod.rs +54 -52
  111. package/rust/core/parser/handler/identifier/on.rs +98 -0
  112. package/rust/core/parser/handler/identifier/print.rs +52 -0
  113. package/rust/core/parser/handler/identifier/sleep.rs +36 -33
  114. package/rust/core/parser/handler/identifier/spawn.rs +91 -88
  115. package/rust/core/parser/handler/identifier/synth.rs +65 -65
  116. package/rust/core/parser/handler/loop_.rs +170 -72
  117. package/rust/core/parser/handler/mod.rs +8 -8
  118. package/rust/core/parser/handler/tempo.rs +53 -47
  119. package/rust/core/parser/mod.rs +4 -4
  120. package/rust/core/parser/statement.rs +142 -108
  121. package/rust/core/plugin/loader.rs +123 -48
  122. package/rust/core/plugin/mod.rs +2 -1
  123. package/rust/core/plugin/runner.rs +296 -0
  124. package/rust/core/preprocessor/loader.rs +515 -326
  125. package/rust/core/preprocessor/mod.rs +4 -4
  126. package/rust/core/preprocessor/module.rs +60 -58
  127. package/rust/core/preprocessor/processor.rs +99 -101
  128. package/rust/core/preprocessor/resolver/bank.rs +51 -49
  129. package/rust/core/preprocessor/resolver/call.rs +100 -100
  130. package/rust/core/preprocessor/resolver/condition.rs +97 -97
  131. package/rust/core/preprocessor/resolver/driver.rs +310 -278
  132. package/rust/core/preprocessor/resolver/function.rs +69 -78
  133. package/rust/core/preprocessor/resolver/group.rs +96 -91
  134. package/rust/core/preprocessor/resolver/let_.rs +32 -28
  135. package/rust/core/preprocessor/resolver/loop_.rs +320 -91
  136. package/rust/core/preprocessor/resolver/mod.rs +15 -15
  137. package/rust/core/preprocessor/resolver/spawn.rs +76 -92
  138. package/rust/core/preprocessor/resolver/synth.rs +56 -50
  139. package/rust/core/preprocessor/resolver/tempo.rs +50 -49
  140. package/rust/core/preprocessor/resolver/trigger.rs +113 -116
  141. package/rust/core/preprocessor/resolver/value.rs +81 -87
  142. package/rust/core/shared/bank.rs +1 -1
  143. package/rust/core/shared/duration.rs +9 -9
  144. package/rust/core/shared/mod.rs +3 -3
  145. package/rust/core/shared/value.rs +35 -32
  146. package/rust/core/store/function.rs +34 -34
  147. package/rust/core/store/global.rs +55 -38
  148. package/rust/core/store/mod.rs +5 -5
  149. package/rust/core/store/variable.rs +37 -34
  150. package/rust/core/utils/mod.rs +2 -2
  151. package/rust/core/utils/path.rs +37 -31
  152. package/rust/core/utils/validation.rs +35 -37
  153. package/rust/installer/addon.rs +84 -80
  154. package/rust/installer/bank.rs +62 -65
  155. package/rust/installer/mod.rs +5 -5
  156. package/rust/installer/plugin.rs +54 -55
  157. package/rust/installer/utils.rs +56 -56
  158. package/rust/lib.rs +156 -164
  159. package/rust/main.rs +250 -145
  160. package/rust/utils/error.rs +200 -0
  161. package/rust/utils/file.rs +38 -35
  162. package/rust/utils/first_usage.rs +76 -0
  163. package/rust/utils/logger.rs +195 -139
  164. package/rust/utils/mod.rs +9 -50
  165. package/rust/utils/signature.rs +19 -17
  166. package/rust/utils/spinner.rs +22 -19
  167. package/rust/utils/telemetry.rs +292 -0
  168. package/rust/utils/watcher.rs +34 -33
  169. package/templates/minimal/README.md +97 -121
  170. package/templates/welcome/README.md +97 -121
  171. package/typescript/bin/index.ts +19 -5
  172. package/typescript/index.ts +3 -1
  173. package/typescript/scripts/postbuild.ts +10 -6
  174. package/typescript/scripts/postinstall.ts +56 -0
  175. package/typescript/scripts/version/bump.ts +0 -1
  176. package/typescript/scripts/version/index.ts +0 -1
  177. package/out-tsc/bin/devalang.exe +0 -0
package/rust/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
- 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
+ }