@devaloop/devalang 0.0.1-alpha.9 → 0.0.1-beta.1

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 (271) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +10 -4
  3. package/.github/workflows/ci.yml +103 -0
  4. package/Cargo.toml +80 -48
  5. package/README.md +135 -154
  6. package/docs/CHANGELOG.md +386 -1
  7. package/docs/CONTRIBUTING.md +101 -0
  8. package/docs/ROADMAP.md +10 -7
  9. package/docs/TODO.md +21 -9
  10. package/examples/automation.deva +42 -0
  11. package/examples/bank.deva +7 -0
  12. package/examples/duration.deva +9 -0
  13. package/examples/events.deva +12 -0
  14. package/examples/function.deva +15 -0
  15. package/examples/index.deva +57 -12
  16. package/examples/loop.deva +5 -12
  17. package/examples/pattern.deva +8 -0
  18. package/examples/plugin.deva +16 -0
  19. package/examples/variables.deva +1 -1
  20. package/out-tsc/bin/index.d.ts +2 -0
  21. package/out-tsc/bin/index.js +51 -7
  22. package/out-tsc/core/functions/index.d.ts +37 -0
  23. package/out-tsc/core/functions/index.js +76 -0
  24. package/out-tsc/core/index.d.ts +6 -0
  25. package/out-tsc/core/index.js +22 -0
  26. package/out-tsc/core/types/index.d.ts +4 -0
  27. package/out-tsc/core/types/index.js +20 -0
  28. package/out-tsc/core/types/plugin.d.ts +18 -0
  29. package/out-tsc/core/types/plugin.js +2 -0
  30. package/out-tsc/core/types/result.d.ts +27 -0
  31. package/out-tsc/core/types/result.js +2 -0
  32. package/out-tsc/core/types/statement.d.ts +106 -0
  33. package/out-tsc/core/types/statement.js +2 -0
  34. package/out-tsc/core/types/value.d.ts +43 -0
  35. package/out-tsc/core/types/value.js +2 -0
  36. package/out-tsc/index.d.ts +7 -0
  37. package/out-tsc/index.js +42 -1
  38. package/out-tsc/pkg/devalang_core.d.ts +13 -0
  39. package/out-tsc/pkg/devalang_core.js +50 -0
  40. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +33 -0
  41. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  42. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  43. package/out-tsc/scripts/postinstall.d.ts +1 -0
  44. package/out-tsc/scripts/postinstall.js +83 -0
  45. package/out-tsc/scripts/version/bump.d.ts +1 -0
  46. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  47. package/out-tsc/scripts/version/index.d.ts +1 -0
  48. package/out-tsc/scripts/version/sync.d.ts +1 -0
  49. package/package.json +28 -7
  50. package/project-version.json +4 -4
  51. package/rust/cli/bank/api.rs +122 -0
  52. package/rust/cli/bank/commands.rs +275 -0
  53. package/rust/cli/bank/mod.rs +29 -0
  54. package/rust/cli/build/commands.rs +103 -0
  55. package/rust/cli/build/mod.rs +2 -0
  56. package/rust/cli/build/process.rs +146 -0
  57. package/rust/cli/check/mod.rs +208 -0
  58. package/rust/cli/discover/commands.rs +253 -0
  59. package/rust/cli/discover/config.rs +111 -0
  60. package/rust/cli/discover/fs.rs +19 -0
  61. package/rust/cli/discover/install.rs +103 -0
  62. package/rust/cli/discover/metadata.rs +48 -0
  63. package/rust/cli/discover/mod.rs +5 -0
  64. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  65. package/rust/cli/init/mod.rs +1 -0
  66. package/rust/cli/install/addon.rs +118 -0
  67. package/rust/cli/install/bank.rs +53 -0
  68. package/rust/cli/install/commands.rs +35 -0
  69. package/rust/cli/install/mod.rs +4 -0
  70. package/rust/cli/install/plugin.rs +61 -0
  71. package/rust/cli/login/commands.rs +124 -0
  72. package/rust/cli/login/mod.rs +1 -0
  73. package/rust/cli/mod.rs +12 -205
  74. package/rust/cli/parser.rs +314 -0
  75. package/rust/cli/play/commands.rs +324 -0
  76. package/rust/cli/play/io.rs +17 -0
  77. package/rust/cli/play/mod.rs +5 -0
  78. package/rust/cli/play/process.rs +150 -0
  79. package/rust/cli/play/realtime.rs +91 -0
  80. package/rust/cli/play/utils.rs +23 -0
  81. package/rust/cli/telemetry/commands.rs +22 -0
  82. package/rust/cli/telemetry/event_creator.rs +80 -0
  83. package/rust/cli/telemetry/mod.rs +3 -0
  84. package/rust/cli/telemetry/send.rs +51 -0
  85. package/rust/cli/{template.rs → template/commands.rs} +69 -57
  86. package/rust/cli/template/mod.rs +1 -0
  87. package/rust/cli/update/commands.rs +6 -0
  88. package/rust/cli/update/mod.rs +1 -0
  89. package/rust/config/driver.rs +103 -0
  90. package/rust/config/mod.rs +3 -16
  91. package/rust/config/ops.rs +26 -0
  92. package/rust/config/settings.rs +101 -0
  93. package/rust/core/audio/engine/helpers.rs +170 -0
  94. package/rust/core/audio/engine/mod.rs +7 -0
  95. package/rust/core/audio/engine/sample.rs +366 -0
  96. package/rust/core/audio/engine/synth.rs +325 -0
  97. package/rust/core/audio/evaluator.rs +310 -31
  98. package/rust/core/audio/interpreter/arrow_call.rs +311 -129
  99. package/rust/core/audio/interpreter/automate.rs +18 -0
  100. package/rust/core/audio/interpreter/call.rs +294 -64
  101. package/rust/core/audio/interpreter/condition.rs +71 -69
  102. package/rust/core/audio/interpreter/driver.rs +542 -216
  103. package/rust/core/audio/interpreter/function.rs +26 -0
  104. package/rust/core/audio/interpreter/let_.rs +38 -19
  105. package/rust/core/audio/interpreter/load.rs +19 -18
  106. package/rust/core/audio/interpreter/loop_.rs +114 -67
  107. package/rust/core/audio/interpreter/mod.rs +14 -12
  108. package/rust/core/audio/interpreter/sleep.rs +28 -36
  109. package/rust/core/audio/interpreter/spawn.rs +252 -66
  110. package/rust/core/audio/interpreter/tempo.rs +40 -16
  111. package/rust/core/audio/interpreter/trigger.rs +239 -69
  112. package/rust/core/audio/loader/mod.rs +1 -1
  113. package/rust/core/audio/loader/trigger.rs +97 -52
  114. package/rust/core/audio/mod.rs +7 -6
  115. package/rust/core/audio/player.rs +70 -54
  116. package/rust/core/audio/renderer.rs +54 -54
  117. package/rust/core/audio/special/easing.rs +189 -0
  118. package/rust/core/audio/special/env.rs +45 -0
  119. package/rust/core/audio/special/math.rs +134 -0
  120. package/rust/core/audio/special/mod.rs +9 -0
  121. package/rust/core/audio/special/modulator.rs +143 -0
  122. package/rust/core/builder/mod.rs +86 -80
  123. package/rust/core/debugger/lexer.rs +27 -27
  124. package/rust/core/debugger/mod.rs +30 -21
  125. package/rust/core/debugger/module.rs +55 -0
  126. package/rust/core/debugger/preprocessor.rs +27 -27
  127. package/rust/core/debugger/store.rs +40 -25
  128. package/rust/core/error/mod.rs +269 -60
  129. package/rust/core/lexer/driver.rs +61 -0
  130. package/rust/core/lexer/handler/arrow.rs +82 -31
  131. package/rust/core/lexer/handler/at.rs +21 -21
  132. package/rust/core/lexer/handler/brace.rs +41 -41
  133. package/rust/core/lexer/handler/colon.rs +21 -21
  134. package/rust/core/lexer/handler/comment.rs +30 -30
  135. package/rust/core/lexer/handler/dot.rs +21 -21
  136. package/rust/core/lexer/handler/driver.rs +337 -226
  137. package/rust/core/lexer/handler/identifier.rs +47 -41
  138. package/rust/core/lexer/handler/indent.rs +66 -52
  139. package/rust/core/lexer/handler/mod.rs +15 -14
  140. package/rust/core/lexer/handler/newline.rs +23 -23
  141. package/rust/core/lexer/handler/number.rs +31 -31
  142. package/rust/core/lexer/handler/operator.rs +46 -44
  143. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  144. package/rust/core/lexer/handler/slash.rs +21 -0
  145. package/rust/core/lexer/handler/string.rs +63 -63
  146. package/rust/core/lexer/mod.rs +3 -51
  147. package/rust/core/lexer/token.rs +17 -12
  148. package/rust/core/mod.rs +10 -10
  149. package/rust/core/parser/driver.rs +584 -331
  150. package/rust/core/parser/handler/arrow_call.rs +253 -126
  151. package/rust/core/parser/handler/at.rs +279 -162
  152. package/rust/core/parser/handler/bank.rs +104 -41
  153. package/rust/core/parser/handler/condition.rs +83 -74
  154. package/rust/core/parser/handler/dot.rs +148 -112
  155. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  156. package/rust/core/parser/handler/identifier/call.rs +91 -41
  157. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  158. package/rust/core/parser/handler/identifier/function.rs +113 -0
  159. package/rust/core/parser/handler/identifier/group.rs +89 -75
  160. package/rust/core/parser/handler/identifier/let_.rs +173 -133
  161. package/rust/core/parser/handler/identifier/mod.rs +55 -51
  162. package/rust/core/parser/handler/identifier/on.rs +107 -0
  163. package/rust/core/parser/handler/identifier/print.rs +49 -0
  164. package/rust/core/parser/handler/identifier/sleep.rs +43 -33
  165. package/rust/core/parser/handler/identifier/spawn.rs +91 -41
  166. package/rust/core/parser/handler/identifier/synth.rs +135 -65
  167. package/rust/core/parser/handler/loop_.rs +194 -72
  168. package/rust/core/parser/handler/mod.rs +9 -8
  169. package/rust/core/parser/handler/pattern.rs +74 -0
  170. package/rust/core/parser/handler/tempo.rs +57 -47
  171. package/rust/core/parser/mod.rs +3 -4
  172. package/rust/core/parser/statement.rs +11 -96
  173. package/rust/core/plugin/loader.rs +137 -0
  174. package/rust/core/plugin/mod.rs +2 -0
  175. package/rust/core/plugin/runner.rs +347 -0
  176. package/rust/core/preprocessor/loader.rs +637 -193
  177. package/rust/core/preprocessor/mod.rs +4 -4
  178. package/rust/core/preprocessor/module.rs +60 -50
  179. package/rust/core/preprocessor/processor.rs +114 -76
  180. package/rust/core/preprocessor/resolver/bank.rs +49 -47
  181. package/rust/core/preprocessor/resolver/call.rs +124 -123
  182. package/rust/core/preprocessor/resolver/condition.rs +95 -92
  183. package/rust/core/preprocessor/resolver/driver.rs +324 -227
  184. package/rust/core/preprocessor/resolver/function.rs +69 -0
  185. package/rust/core/preprocessor/resolver/group.rs +94 -61
  186. package/rust/core/preprocessor/resolver/let_.rs +32 -31
  187. package/rust/core/preprocessor/resolver/loop_.rs +318 -91
  188. package/rust/core/preprocessor/resolver/mod.rs +16 -14
  189. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  190. package/rust/core/preprocessor/resolver/spawn.rs +99 -58
  191. package/rust/core/preprocessor/resolver/synth.rs +54 -50
  192. package/rust/core/preprocessor/resolver/tempo.rs +48 -49
  193. package/rust/core/preprocessor/resolver/trigger.rs +116 -112
  194. package/rust/core/preprocessor/resolver/value.rs +176 -78
  195. package/rust/core/store/export.rs +28 -28
  196. package/rust/core/store/function.rs +40 -0
  197. package/rust/core/store/global.rs +61 -39
  198. package/rust/core/store/import.rs +28 -28
  199. package/rust/core/store/mod.rs +5 -4
  200. package/rust/core/store/variable.rs +51 -28
  201. package/rust/core/utils/mod.rs +1 -2
  202. package/rust/core/utils/path.rs +37 -31
  203. package/rust/lib.rs +308 -117
  204. package/rust/main.rs +364 -65
  205. package/rust/types/Cargo.toml +11 -0
  206. package/rust/types/src/addons.rs +55 -0
  207. package/rust/types/src/ast.rs +202 -0
  208. package/rust/types/src/config.rs +74 -0
  209. package/rust/types/src/lib.rs +12 -0
  210. package/rust/types/src/telemetry.rs +85 -0
  211. package/rust/utils/Cargo.toml +26 -0
  212. package/rust/utils/src/error.rs +186 -0
  213. package/rust/utils/src/file.rs +94 -0
  214. package/rust/utils/src/first_usage.rs +97 -0
  215. package/rust/utils/{mod.rs → src/lib.rs} +9 -6
  216. package/rust/utils/{logger.rs → src/logger.rs} +200 -123
  217. package/rust/utils/src/path.rs +88 -0
  218. package/rust/utils/src/signature.rs +41 -0
  219. package/rust/utils/{spinner.rs → src/spinner.rs} +20 -21
  220. package/rust/utils/src/version.rs +27 -0
  221. package/rust/utils/{watcher.rs → src/watcher.rs} +46 -33
  222. package/rust/web/api.rs +5 -0
  223. package/rust/web/cdn.rs +34 -0
  224. package/rust/web/mod.rs +3 -0
  225. package/rust/web/sso.rs +5 -0
  226. package/templates/minimal/README.md +143 -127
  227. package/templates/welcome/README.md +143 -127
  228. package/templates/welcome/src/index.deva +56 -8
  229. package/templates/welcome/src/variables.deva +2 -4
  230. package/tests/integration.rs +21 -0
  231. package/tests/rust/cli_check_build.rs +21 -0
  232. package/tests/rust/cli_help.rs +12 -0
  233. package/tests/rust/cli_template_list.rs +10 -0
  234. package/tests/rust/cli_version.rs +11 -0
  235. package/tests/typescript/index.spec.ts +136 -0
  236. package/tests/typescript/playhead.spec.ts +36 -0
  237. package/tests/typescript/render_e2e.spec.ts +77 -0
  238. package/tsconfig.json +12 -10
  239. package/typescript/bin/index.ts +19 -5
  240. package/typescript/core/functions/index.ts +83 -0
  241. package/typescript/core/index.ts +6 -0
  242. package/typescript/core/types/index.ts +4 -0
  243. package/typescript/core/types/plugin.ts +19 -0
  244. package/typescript/core/types/result.ts +29 -0
  245. package/typescript/core/types/statement.ts +47 -0
  246. package/typescript/core/types/value.ts +29 -0
  247. package/typescript/index.ts +8 -1
  248. package/typescript/pkg/devalang_core.d.ts +4 -0
  249. package/typescript/pkg/devalang_core.ts +49 -0
  250. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  251. package/typescript/scripts/postinstall.ts +85 -0
  252. package/typescript/scripts/version/bump.ts +0 -1
  253. package/typescript/scripts/version/index.ts +0 -1
  254. package/docs/COMMANDS.md +0 -85
  255. package/docs/CONFIG.md +0 -30
  256. package/docs/SYNTAX.md +0 -210
  257. package/out-tsc/bin/devalang.exe +0 -0
  258. package/out-tsc/scripts/postbuild.js +0 -11
  259. package/rust/cli/build.rs +0 -137
  260. package/rust/cli/check.rs +0 -117
  261. package/rust/cli/play.rs +0 -193
  262. package/rust/config/loader.rs +0 -13
  263. package/rust/core/audio/engine.rs +0 -203
  264. package/rust/core/shared/duration.rs +0 -8
  265. package/rust/core/shared/mod.rs +0 -2
  266. package/rust/core/shared/value.rs +0 -18
  267. package/rust/core/utils/validation.rs +0 -37
  268. package/rust/utils/file.rs +0 -35
  269. package/rust/utils/signature.rs +0 -17
  270. package/rust/utils/version.rs +0 -15
  271. package/typescript/scripts/postbuild.ts +0 -8
@@ -0,0 +1,80 @@
1
+ use crate::config::settings::get_user_config;
2
+ use devalang_types::{
3
+ TelemetryErrorLevel as SharedTelemetryErrorLevel, TelemetryEvent as SharedTelemetryEvent,
4
+ };
5
+ use uuid::Uuid;
6
+
7
+ pub type TelemetryEvent = SharedTelemetryEvent;
8
+ pub type TelemetryErrorLevel = SharedTelemetryErrorLevel;
9
+
10
+ pub trait TelemetryEventExt {
11
+ fn set_timestamp(&mut self, timestamp: String);
12
+ fn set_duration(&mut self, duration: u64);
13
+ fn set_success(&mut self, success: bool);
14
+ fn set_error(
15
+ &mut self,
16
+ level: TelemetryErrorLevel,
17
+ message: Option<String>,
18
+ exit_code: Option<i32>,
19
+ );
20
+ }
21
+
22
+ impl TelemetryEventExt for SharedTelemetryEvent {
23
+ fn set_timestamp(&mut self, timestamp: String) {
24
+ self.timestamp = timestamp;
25
+ }
26
+
27
+ fn set_duration(&mut self, duration: u64) {
28
+ self.duration = duration;
29
+ }
30
+
31
+ fn set_success(&mut self, success: bool) {
32
+ self.success = success;
33
+ }
34
+
35
+ fn set_error(
36
+ &mut self,
37
+ level: TelemetryErrorLevel,
38
+ message: Option<String>,
39
+ exit_code: Option<i32>,
40
+ ) {
41
+ self.error_level = level;
42
+ self.error_message = message;
43
+ self.exit_code = exit_code;
44
+ }
45
+ }
46
+
47
+ pub struct TelemetryEventCreator {
48
+ pub events: Vec<TelemetryEvent>,
49
+ }
50
+
51
+ impl TelemetryEventCreator {
52
+ pub fn new() -> Self {
53
+ TelemetryEventCreator { events: Vec::new() }
54
+ }
55
+
56
+ pub fn create_event(&mut self, event: TelemetryEvent) {
57
+ self.events.push(event.clone());
58
+ }
59
+
60
+ pub fn get_base_event(&self) -> TelemetryEvent {
61
+ let uuid = match get_user_config() {
62
+ Some(cfg) if !cfg.telemetry.uuid.is_empty() => cfg.telemetry.uuid.clone(),
63
+ _ => Uuid::new_v4().to_string(),
64
+ };
65
+
66
+ TelemetryEvent {
67
+ uuid,
68
+ cli_version: env!("CARGO_PKG_VERSION").to_string(),
69
+ os: std::env::consts::OS.to_string(),
70
+ command: std::env::args().collect::<Vec<_>>(),
71
+ project_info: None,
72
+ error_level: TelemetryErrorLevel::None,
73
+ error_message: None,
74
+ exit_code: None,
75
+ timestamp: chrono::Utc::now().to_string(),
76
+ duration: 0,
77
+ success: true,
78
+ }
79
+ }
80
+ }
@@ -0,0 +1,3 @@
1
+ pub mod commands;
2
+ pub mod event_creator;
3
+ pub mod send;
@@ -0,0 +1,51 @@
1
+ use std::time::Duration;
2
+
3
+ use crate::{config::settings::get_user_config, web::api::get_api_url};
4
+ use devalang_types::{TelemetryEvent, TelemetrySendError};
5
+
6
+ pub async fn send_telemetry_event(event: &TelemetryEvent) -> Result<(), TelemetrySendError> {
7
+ if let Some(cfg) = get_user_config() {
8
+ if cfg.telemetry.enabled == false {
9
+ return Ok(());
10
+ }
11
+ } else {
12
+ return Ok(());
13
+ }
14
+
15
+ let telemetry_url = format!("{}/v1/telemetry/send", get_api_url());
16
+ let client = reqwest::Client::builder()
17
+ .timeout(Duration::from_secs(5))
18
+ .build()
19
+ .map_err(|e| TelemetrySendError::Http(format!("client build error: {}", e)))?;
20
+
21
+ let mut last_err: Option<String> = None;
22
+ for (i, delay_ms) in [0u64, 250, 500, 1000].iter().enumerate() {
23
+ if *delay_ms > 0 {
24
+ tokio::time::sleep(Duration::from_millis(*delay_ms)).await;
25
+ }
26
+
27
+ let res = client
28
+ .post(telemetry_url.clone())
29
+ .json(event)
30
+ .send()
31
+ .await
32
+ .and_then(|r| r.error_for_status());
33
+
34
+ match res {
35
+ Ok(_) => {
36
+ return Ok(());
37
+ }
38
+ Err(err) => {
39
+ last_err = Some(err.to_string());
40
+
41
+ if i == 3 {
42
+ break;
43
+ }
44
+ }
45
+ }
46
+ }
47
+
48
+ Err(TelemetrySendError::Http(
49
+ last_err.unwrap_or_else(|| "unknown error".to_string()),
50
+ ))
51
+ }
@@ -1,57 +1,69 @@
1
- use include_dir::{ include_dir, Dir, DirEntry };
2
- use crate::utils::file::format_file_size;
3
-
4
- static TEMPLATES_DIR: Dir = include_dir!("$CARGO_MANIFEST_DIR/templates");
5
-
6
- #[cfg(feature = "cli")]
7
- pub fn handle_template_list_command() {
8
- let available_templates = get_available_templates();
9
-
10
- println!("📦 Available templates ({}) :\n", available_templates.len());
11
-
12
- for dir in available_templates {
13
- println!("• {}", dir);
14
- }
15
-
16
- println!("\nUsage : devalang init --name <project-name> --template <template-name>");
17
- }
18
-
19
- #[cfg(feature = "cli")]
20
- pub fn handle_template_info_command(name: String) {
21
- let template_dir = TEMPLATES_DIR.get_dir(name.clone()).unwrap_or_else(|| {
22
- println!("❌ The template '{}' is not found.", name);
23
-
24
- std::process::exit(1);
25
- });
26
-
27
- let mut file_count = 0;
28
- let mut dir_count = 0;
29
- let mut total_size: u64 = 0;
30
-
31
- fn walk(dir: &Dir, file_count: &mut u32, dir_count: &mut u32, total_size: &mut u64) {
32
- for entry in dir.entries() {
33
- match entry {
34
- DirEntry::File(file) => {
35
- *file_count += 1;
36
- *total_size += file.contents().len() as u64;
37
- }
38
- DirEntry::Dir(subdir) => {
39
- *dir_count += 1;
40
- walk(subdir, file_count, dir_count, total_size);
41
- }
42
- }
43
- }
44
- }
45
-
46
- walk(template_dir, &mut file_count, &mut dir_count, &mut total_size);
47
-
48
- println!("📦 Template : {}", name);
49
- println!("📂 Content : {file_count} file(s), {dir_count} folder(s)");
50
- println!("💾 Size : {}", format_file_size(total_size));
51
- }
52
-
53
- pub fn get_available_templates() -> Vec<String> {
54
- TEMPLATES_DIR.dirs()
55
- .map(|dir| dir.path().file_name().unwrap().to_string_lossy().to_string())
56
- .collect()
57
- }
1
+ use devalang_utils::file::format_file_size;
2
+ use include_dir::{Dir, DirEntry, include_dir};
3
+
4
+ static TEMPLATES_DIR: Dir = include_dir!("$CARGO_MANIFEST_DIR/templates");
5
+
6
+ #[cfg(feature = "cli")]
7
+ pub fn handle_template_list_command() {
8
+ let available_templates = get_available_templates();
9
+
10
+ println!("📦 Available templates ({}) :\n", available_templates.len());
11
+
12
+ for dir in available_templates {
13
+ println!("• {}", dir);
14
+ }
15
+
16
+ println!("\nUsage : devalang init --name <project-name> --template <template-name>");
17
+ }
18
+
19
+ #[cfg(feature = "cli")]
20
+ pub fn handle_template_info_command(name: String) {
21
+ let template_dir = TEMPLATES_DIR.get_dir(name.clone()).unwrap_or_else(|| {
22
+ println!("❌ The template '{}' is not found.", name);
23
+
24
+ std::process::exit(1);
25
+ });
26
+
27
+ let mut file_count = 0;
28
+ let mut dir_count = 0;
29
+ let mut total_size: u64 = 0;
30
+
31
+ fn walk(dir: &Dir, file_count: &mut u32, dir_count: &mut u32, total_size: &mut u64) {
32
+ for entry in dir.entries() {
33
+ match entry {
34
+ DirEntry::File(file) => {
35
+ *file_count += 1;
36
+ *total_size += file.contents().len() as u64;
37
+ }
38
+ DirEntry::Dir(subdir) => {
39
+ *dir_count += 1;
40
+ walk(subdir, file_count, dir_count, total_size);
41
+ }
42
+ }
43
+ }
44
+ }
45
+
46
+ walk(
47
+ template_dir,
48
+ &mut file_count,
49
+ &mut dir_count,
50
+ &mut total_size,
51
+ );
52
+
53
+ println!("📦 Template : {}", name);
54
+ println!("📂 Content : {file_count} file(s), {dir_count} folder(s)");
55
+ println!("💾 Size : {}", format_file_size(total_size));
56
+ }
57
+
58
+ pub fn get_available_templates() -> Vec<String> {
59
+ TEMPLATES_DIR
60
+ .dirs()
61
+ .map(|dir| {
62
+ dir.path()
63
+ .file_name()
64
+ .unwrap()
65
+ .to_string_lossy()
66
+ .to_string()
67
+ })
68
+ .collect()
69
+ }
@@ -0,0 +1 @@
1
+ pub mod commands;
@@ -0,0 +1,6 @@
1
+ pub async fn handle_update_command(
2
+ _only: Option<String>,
3
+ ) -> Result<(), Box<dyn std::error::Error>> {
4
+ println!("Updates are not yet implemented.");
5
+ Ok(())
6
+ }
@@ -0,0 +1 @@
1
+ pub mod commands;
@@ -0,0 +1,103 @@
1
+ use devalang_types::{
2
+ PluginEntry as SharedPluginEntry, ProjectConfig as SharedProjectConfig,
3
+ ProjectConfigBankEntry as SharedProjectConfigBankEntry,
4
+ ProjectConfigDefaults as SharedProjectConfigDefaults,
5
+ ProjectConfigPluginEntry as SharedProjectConfigPluginEntry,
6
+ };
7
+ use devalang_utils::path as path_utils;
8
+ use std::path::PathBuf;
9
+
10
+ pub type ProjectConfig = SharedProjectConfig;
11
+ pub type ProjectConfigDefaults = SharedProjectConfigDefaults;
12
+ pub type ProjectConfigBankEntry = SharedProjectConfigBankEntry;
13
+ pub type ProjectConfigPluginEntry = SharedProjectConfigPluginEntry;
14
+ pub type PluginEntry = SharedPluginEntry;
15
+
16
+ pub trait ProjectConfigExt {
17
+ fn new_config() -> Self;
18
+ fn with_defaults(
19
+ entry: Option<String>,
20
+ output: Option<String>,
21
+ watch: Option<bool>,
22
+ repeat: Option<bool>,
23
+ debug: Option<bool>,
24
+ compress: Option<bool>,
25
+ ) -> Self;
26
+ fn get() -> Result<Self, String>
27
+ where
28
+ Self: Sized;
29
+ fn from_string(config_string: &str) -> Result<(Self, String), String>
30
+ where
31
+ Self: Sized;
32
+ fn write_config(&self, new_config: &Self) -> Result<(), String>
33
+ where
34
+ Self: Sized;
35
+ }
36
+
37
+ impl ProjectConfigExt for SharedProjectConfig {
38
+ fn new_config() -> Self {
39
+ SharedProjectConfig::default()
40
+ }
41
+
42
+ fn with_defaults(
43
+ entry: Option<String>,
44
+ output: Option<String>,
45
+ watch: Option<bool>,
46
+ repeat: Option<bool>,
47
+ debug: Option<bool>,
48
+ compress: Option<bool>,
49
+ ) -> Self {
50
+ SharedProjectConfig {
51
+ defaults: SharedProjectConfigDefaults {
52
+ entry,
53
+ output,
54
+ watch,
55
+ repeat,
56
+ debug,
57
+ compress,
58
+ },
59
+ banks: Some(Vec::new()),
60
+ plugins: Some(Vec::new()),
61
+ }
62
+ }
63
+
64
+ fn get() -> Result<Self, String> {
65
+ let config_path = path_utils::get_devalang_config_path()?;
66
+
67
+ let config_content = std::fs::read_to_string(&config_path)
68
+ .map_err(|e| format!("Failed to read config file: {}", e))?;
69
+
70
+ let config: SharedProjectConfig = toml::from_str(&config_content)
71
+ .map_err(|e| format!("Failed to parse config file: {}", e))?;
72
+
73
+ Ok(config)
74
+ }
75
+
76
+ fn from_string(config_string: &str) -> Result<(Self, String), String> {
77
+ let config: SharedProjectConfig = toml::from_str(config_string)
78
+ .map_err(|e| format!("Failed to parse config string: {}", e))?;
79
+ let config_path = ".devalang".to_string();
80
+
81
+ Ok((config, config_path))
82
+ }
83
+
84
+ fn write_config(&self, new_config: &Self) -> Result<(), String> {
85
+ let config_path: PathBuf = match path_utils::get_project_root() {
86
+ Ok(root) => root.join(path_utils::DEVALANG_CONFIG),
87
+ Err(_) => PathBuf::from(path_utils::DEVALANG_CONFIG),
88
+ };
89
+
90
+ let content = toml::to_string(new_config)
91
+ .map_err(|e| format!("Failed to serialize config: {}", e))?;
92
+
93
+ std::fs::write(&config_path, content).map_err(|e| {
94
+ format!(
95
+ "Failed to write config to file '{}': {}",
96
+ config_path.display(),
97
+ e
98
+ )
99
+ })?;
100
+
101
+ Ok(())
102
+ }
103
+ }
@@ -1,16 +1,3 @@
1
- pub mod loader;
2
-
3
- use serde::Deserialize;
4
-
5
- #[derive(Debug, Deserialize, Clone)]
6
- pub struct Config {
7
- pub defaults: ConfigDefaults,
8
- }
9
-
10
- #[derive(Debug, Deserialize, Clone)]
11
- pub struct ConfigDefaults {
12
- pub entry: Option<String>,
13
- pub output: Option<String>,
14
- pub watch: Option<bool>,
15
- pub repeat: Option<bool>,
16
- }
1
+ pub mod driver;
2
+ pub mod ops;
3
+ pub mod settings;
@@ -0,0 +1,26 @@
1
+ use crate::config::driver::ProjectConfig;
2
+ use std::fs;
3
+ use std::path::Path;
4
+
5
+ pub fn load_config(path: Option<&Path>) -> Option<ProjectConfig> {
6
+ let config_path_buf;
7
+ let config_path = match path {
8
+ Some(p) => p,
9
+ None => {
10
+ config_path_buf = match devalang_utils::path::get_devalang_config_path() {
11
+ Ok(p) => p,
12
+ Err(_) => {
13
+ return None;
14
+ }
15
+ };
16
+ &config_path_buf
17
+ }
18
+ };
19
+
20
+ if config_path.exists() {
21
+ let content = fs::read_to_string(config_path).ok()?;
22
+ toml::from_str(&content).ok()
23
+ } else {
24
+ None
25
+ }
26
+ }
@@ -0,0 +1,101 @@
1
+ use devalang_types::{TelemetrySettings, UserSettings};
2
+ use serde_json::Value as JsonValue;
3
+ use std::io::Write;
4
+
5
+ pub fn get_home_dir() -> Option<std::path::PathBuf> {
6
+ dirs::home_dir()
7
+ }
8
+
9
+ pub fn get_devalang_homedir() -> std::path::PathBuf {
10
+ if let Some(home_dir) = get_home_dir() {
11
+ home_dir.join(".devalang")
12
+ } else {
13
+ std::path::PathBuf::from("~/.devalang")
14
+ }
15
+ }
16
+
17
+ pub fn get_default_user_config() -> UserSettings {
18
+ UserSettings {
19
+ session: "".into(),
20
+ telemetry: TelemetrySettings {
21
+ uuid: uuid::Uuid::new_v4().to_string(),
22
+ enabled: false,
23
+ level: "basic".into(),
24
+ stats: false,
25
+ },
26
+ }
27
+ }
28
+
29
+ pub fn get_user_config() -> Option<UserSettings> {
30
+ if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
31
+ let file = std::fs::File::open(config_path).ok()?;
32
+ let settings = serde_json::from_reader(file).ok()?;
33
+ Some(settings)
34
+ } else {
35
+ None
36
+ }
37
+ }
38
+
39
+ pub fn write_user_config_file() {
40
+ if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
41
+ let settings = get_user_config().unwrap_or_else(get_default_user_config);
42
+
43
+ let config_json = serde_json::to_string(&settings).unwrap();
44
+
45
+ if let Err(e) = write_config_atomic(&config_path, &config_json) {
46
+ println!("Could not write config file: {}", e);
47
+ }
48
+ } else {
49
+ println!("Could not create config file");
50
+ }
51
+ }
52
+
53
+ pub fn ensure_user_config_file_exists() {
54
+ if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
55
+ if !config_path.exists() {
56
+ write_user_config_file();
57
+ }
58
+ }
59
+ }
60
+
61
+ pub fn set_user_config_value(key: &str, value: JsonValue) {
62
+ let mut settings = get_user_config().unwrap_or_default();
63
+
64
+ match (key, &value) {
65
+ ("telemetry", JsonValue::Bool(b)) => {
66
+ settings.telemetry.enabled = *b;
67
+ }
68
+ ("session", JsonValue::String(s)) => {
69
+ settings.session = s.clone();
70
+ }
71
+ _ => {
72
+ println!("Unsupported key or value type for '{}': {:?}", key, value);
73
+ }
74
+ }
75
+
76
+ if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
77
+ let config_json = serde_json::to_string(&settings).unwrap();
78
+ if let Err(e) = write_config_atomic(&config_path, &config_json) {
79
+ println!("Could not write config file: {}", e);
80
+ }
81
+ } else {
82
+ println!("Could not create config file");
83
+ }
84
+ }
85
+
86
+ pub fn write_config_atomic(
87
+ config_path: &std::path::PathBuf,
88
+ contents: &str,
89
+ ) -> std::io::Result<()> {
90
+ if let Some(parent) = config_path.parent() {
91
+ std::fs::create_dir_all(parent)?;
92
+ }
93
+
94
+ let tmp_path = config_path.with_extension("json.tmp");
95
+ let mut tmp_file = std::fs::File::create(&tmp_path)?;
96
+ tmp_file.write_all(contents.as_bytes())?;
97
+ tmp_file.sync_all()?;
98
+ std::fs::rename(&tmp_path, config_path)?;
99
+
100
+ Ok(())
101
+ }