@devaloop/devalang 0.0.1-alpha.8 → 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 (277) 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 -158
  6. package/docs/CHANGELOG.md +413 -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/condition.deva +8 -12
  13. package/examples/duration.deva +9 -0
  14. package/examples/events.deva +12 -0
  15. package/examples/function.deva +15 -0
  16. package/examples/group.deva +3 -3
  17. package/examples/index.deva +57 -10
  18. package/examples/loop.deva +7 -12
  19. package/examples/pattern.deva +8 -0
  20. package/examples/plugin.deva +16 -0
  21. package/examples/synth.deva +14 -0
  22. package/examples/variables.deva +2 -2
  23. package/out-tsc/bin/index.d.ts +2 -0
  24. package/out-tsc/bin/index.js +51 -7
  25. package/out-tsc/core/functions/index.d.ts +37 -0
  26. package/out-tsc/core/functions/index.js +76 -0
  27. package/out-tsc/core/index.d.ts +6 -0
  28. package/out-tsc/core/index.js +22 -0
  29. package/out-tsc/core/types/index.d.ts +4 -0
  30. package/out-tsc/core/types/index.js +20 -0
  31. package/out-tsc/core/types/plugin.d.ts +18 -0
  32. package/out-tsc/core/types/plugin.js +2 -0
  33. package/out-tsc/core/types/result.d.ts +27 -0
  34. package/out-tsc/core/types/result.js +2 -0
  35. package/out-tsc/core/types/statement.d.ts +106 -0
  36. package/out-tsc/core/types/statement.js +2 -0
  37. package/out-tsc/core/types/value.d.ts +43 -0
  38. package/out-tsc/core/types/value.js +2 -0
  39. package/out-tsc/index.d.ts +7 -0
  40. package/out-tsc/index.js +42 -1
  41. package/out-tsc/pkg/devalang_core.d.ts +13 -0
  42. package/out-tsc/pkg/devalang_core.js +50 -0
  43. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +33 -0
  44. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  45. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  46. package/out-tsc/scripts/postinstall.d.ts +1 -0
  47. package/out-tsc/scripts/postinstall.js +83 -0
  48. package/out-tsc/scripts/version/bump.d.ts +1 -0
  49. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  50. package/out-tsc/scripts/version/fetch.js +1 -5
  51. package/out-tsc/scripts/version/index.d.ts +1 -0
  52. package/out-tsc/scripts/version/sync.d.ts +1 -0
  53. package/package.json +28 -7
  54. package/project-version.json +4 -4
  55. package/rust/cli/bank/api.rs +122 -0
  56. package/rust/cli/bank/commands.rs +275 -0
  57. package/rust/cli/bank/mod.rs +29 -0
  58. package/rust/cli/build/commands.rs +103 -0
  59. package/rust/cli/build/mod.rs +2 -0
  60. package/rust/cli/build/process.rs +146 -0
  61. package/rust/cli/check/mod.rs +208 -0
  62. package/rust/cli/discover/commands.rs +253 -0
  63. package/rust/cli/discover/config.rs +111 -0
  64. package/rust/cli/discover/fs.rs +19 -0
  65. package/rust/cli/discover/install.rs +103 -0
  66. package/rust/cli/discover/metadata.rs +48 -0
  67. package/rust/cli/discover/mod.rs +5 -0
  68. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  69. package/rust/cli/init/mod.rs +1 -0
  70. package/rust/cli/install/addon.rs +118 -0
  71. package/rust/cli/install/bank.rs +53 -0
  72. package/rust/cli/install/commands.rs +35 -0
  73. package/rust/cli/install/mod.rs +4 -0
  74. package/rust/cli/install/plugin.rs +61 -0
  75. package/rust/cli/login/commands.rs +124 -0
  76. package/rust/cli/login/mod.rs +1 -0
  77. package/rust/cli/mod.rs +12 -205
  78. package/rust/cli/parser.rs +314 -0
  79. package/rust/cli/play/commands.rs +324 -0
  80. package/rust/cli/play/io.rs +17 -0
  81. package/rust/cli/play/mod.rs +5 -0
  82. package/rust/cli/play/process.rs +150 -0
  83. package/rust/cli/play/realtime.rs +91 -0
  84. package/rust/cli/play/utils.rs +23 -0
  85. package/rust/cli/telemetry/commands.rs +22 -0
  86. package/rust/cli/telemetry/event_creator.rs +80 -0
  87. package/rust/cli/telemetry/mod.rs +3 -0
  88. package/rust/cli/telemetry/send.rs +51 -0
  89. package/rust/cli/{template.rs → template/commands.rs} +69 -57
  90. package/rust/cli/template/mod.rs +1 -0
  91. package/rust/cli/update/commands.rs +6 -0
  92. package/rust/cli/update/mod.rs +1 -0
  93. package/rust/config/driver.rs +103 -0
  94. package/rust/config/mod.rs +3 -16
  95. package/rust/config/ops.rs +26 -0
  96. package/rust/config/settings.rs +101 -0
  97. package/rust/core/audio/engine/helpers.rs +170 -0
  98. package/rust/core/audio/engine/mod.rs +7 -0
  99. package/rust/core/audio/engine/sample.rs +366 -0
  100. package/rust/core/audio/engine/synth.rs +325 -0
  101. package/rust/core/audio/evaluator.rs +310 -31
  102. package/rust/core/audio/interpreter/arrow_call.rs +311 -0
  103. package/rust/core/audio/interpreter/automate.rs +18 -0
  104. package/rust/core/audio/interpreter/call.rs +294 -42
  105. package/rust/core/audio/interpreter/condition.rs +71 -65
  106. package/rust/core/audio/interpreter/driver.rs +542 -204
  107. package/rust/core/audio/interpreter/function.rs +26 -0
  108. package/rust/core/audio/interpreter/let_.rs +38 -19
  109. package/rust/core/audio/interpreter/load.rs +19 -18
  110. package/rust/core/audio/interpreter/loop_.rs +114 -59
  111. package/rust/core/audio/interpreter/mod.rs +14 -11
  112. package/rust/core/audio/interpreter/sleep.rs +28 -36
  113. package/rust/core/audio/interpreter/spawn.rs +252 -65
  114. package/rust/core/audio/interpreter/tempo.rs +40 -16
  115. package/rust/core/audio/interpreter/trigger.rs +239 -69
  116. package/rust/core/audio/loader/mod.rs +1 -1
  117. package/rust/core/audio/loader/trigger.rs +97 -52
  118. package/rust/core/audio/mod.rs +7 -6
  119. package/rust/core/audio/player.rs +70 -54
  120. package/rust/core/audio/renderer.rs +54 -57
  121. package/rust/core/audio/special/easing.rs +189 -0
  122. package/rust/core/audio/special/env.rs +45 -0
  123. package/rust/core/audio/special/math.rs +134 -0
  124. package/rust/core/audio/special/mod.rs +9 -0
  125. package/rust/core/audio/special/modulator.rs +143 -0
  126. package/rust/core/builder/mod.rs +86 -80
  127. package/rust/core/debugger/lexer.rs +27 -27
  128. package/rust/core/debugger/mod.rs +30 -21
  129. package/rust/core/debugger/module.rs +55 -0
  130. package/rust/core/debugger/preprocessor.rs +27 -27
  131. package/rust/core/debugger/store.rs +40 -25
  132. package/rust/core/error/mod.rs +269 -60
  133. package/rust/core/lexer/driver.rs +61 -0
  134. package/rust/core/lexer/handler/arrow.rs +82 -0
  135. package/rust/core/lexer/handler/at.rs +21 -21
  136. package/rust/core/lexer/handler/brace.rs +41 -41
  137. package/rust/core/lexer/handler/colon.rs +21 -21
  138. package/rust/core/lexer/handler/comment.rs +30 -30
  139. package/rust/core/lexer/handler/dot.rs +21 -21
  140. package/rust/core/lexer/handler/driver.rs +337 -215
  141. package/rust/core/lexer/handler/identifier.rs +47 -40
  142. package/rust/core/lexer/handler/indent.rs +66 -52
  143. package/rust/core/lexer/handler/mod.rs +15 -13
  144. package/rust/core/lexer/handler/newline.rs +23 -23
  145. package/rust/core/lexer/handler/number.rs +31 -31
  146. package/rust/core/lexer/handler/operator.rs +46 -44
  147. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  148. package/rust/core/lexer/handler/slash.rs +21 -0
  149. package/rust/core/lexer/handler/string.rs +63 -63
  150. package/rust/core/lexer/mod.rs +3 -30
  151. package/rust/core/lexer/token.rs +21 -12
  152. package/rust/core/mod.rs +10 -10
  153. package/rust/core/parser/driver.rs +584 -312
  154. package/rust/core/parser/handler/arrow_call.rs +253 -0
  155. package/rust/core/parser/handler/at.rs +279 -162
  156. package/rust/core/parser/handler/bank.rs +104 -41
  157. package/rust/core/parser/handler/condition.rs +83 -74
  158. package/rust/core/parser/handler/dot.rs +148 -112
  159. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  160. package/rust/core/parser/handler/identifier/call.rs +91 -0
  161. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  162. package/rust/core/parser/handler/identifier/function.rs +113 -0
  163. package/rust/core/parser/handler/identifier/group.rs +89 -0
  164. package/rust/core/parser/handler/identifier/let_.rs +173 -0
  165. package/rust/core/parser/handler/identifier/mod.rs +55 -0
  166. package/rust/core/parser/handler/identifier/on.rs +107 -0
  167. package/rust/core/parser/handler/identifier/print.rs +49 -0
  168. package/rust/core/parser/handler/identifier/sleep.rs +43 -0
  169. package/rust/core/parser/handler/identifier/spawn.rs +91 -0
  170. package/rust/core/parser/handler/identifier/synth.rs +135 -0
  171. package/rust/core/parser/handler/loop_.rs +194 -66
  172. package/rust/core/parser/handler/mod.rs +9 -7
  173. package/rust/core/parser/handler/pattern.rs +74 -0
  174. package/rust/core/parser/handler/tempo.rs +57 -47
  175. package/rust/core/parser/mod.rs +3 -4
  176. package/rust/core/parser/statement.rs +11 -88
  177. package/rust/core/plugin/loader.rs +137 -0
  178. package/rust/core/plugin/mod.rs +2 -0
  179. package/rust/core/plugin/runner.rs +347 -0
  180. package/rust/core/preprocessor/loader.rs +637 -179
  181. package/rust/core/preprocessor/mod.rs +4 -4
  182. package/rust/core/preprocessor/module.rs +60 -53
  183. package/rust/core/preprocessor/processor.rs +114 -67
  184. package/rust/core/preprocessor/resolver/bank.rs +49 -47
  185. package/rust/core/preprocessor/resolver/call.rs +124 -53
  186. package/rust/core/preprocessor/resolver/condition.rs +95 -66
  187. package/rust/core/preprocessor/resolver/driver.rs +324 -182
  188. package/rust/core/preprocessor/resolver/function.rs +69 -0
  189. package/rust/core/preprocessor/resolver/group.rs +94 -118
  190. package/rust/core/preprocessor/resolver/let_.rs +32 -0
  191. package/rust/core/preprocessor/resolver/loop_.rs +318 -145
  192. package/rust/core/preprocessor/resolver/mod.rs +16 -10
  193. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  194. package/rust/core/preprocessor/resolver/spawn.rs +99 -53
  195. package/rust/core/preprocessor/resolver/synth.rs +54 -0
  196. package/rust/core/preprocessor/resolver/tempo.rs +48 -49
  197. package/rust/core/preprocessor/resolver/trigger.rs +116 -111
  198. package/rust/core/preprocessor/resolver/value.rs +176 -0
  199. package/rust/core/store/export.rs +28 -28
  200. package/rust/core/store/function.rs +40 -0
  201. package/rust/core/store/global.rs +61 -39
  202. package/rust/core/store/import.rs +28 -28
  203. package/rust/core/store/mod.rs +5 -4
  204. package/rust/core/store/variable.rs +51 -28
  205. package/rust/core/utils/mod.rs +1 -2
  206. package/rust/core/utils/path.rs +37 -46
  207. package/rust/lib.rs +308 -117
  208. package/rust/main.rs +364 -65
  209. package/rust/types/Cargo.toml +11 -0
  210. package/rust/types/src/addons.rs +55 -0
  211. package/rust/types/src/ast.rs +202 -0
  212. package/rust/types/src/config.rs +74 -0
  213. package/rust/types/src/lib.rs +12 -0
  214. package/rust/types/src/telemetry.rs +85 -0
  215. package/rust/utils/Cargo.toml +26 -0
  216. package/rust/utils/src/error.rs +186 -0
  217. package/rust/utils/src/file.rs +94 -0
  218. package/rust/utils/src/first_usage.rs +97 -0
  219. package/rust/utils/{mod.rs → src/lib.rs} +9 -6
  220. package/rust/utils/{logger.rs → src/logger.rs} +200 -123
  221. package/rust/utils/src/path.rs +88 -0
  222. package/rust/utils/src/signature.rs +41 -0
  223. package/rust/utils/{spinner.rs → src/spinner.rs} +20 -21
  224. package/rust/utils/src/version.rs +27 -0
  225. package/rust/utils/{watcher.rs → src/watcher.rs} +46 -33
  226. package/rust/web/api.rs +5 -0
  227. package/rust/web/cdn.rs +34 -0
  228. package/rust/web/mod.rs +3 -0
  229. package/rust/web/sso.rs +5 -0
  230. package/templates/minimal/README.md +143 -127
  231. package/templates/welcome/README.md +143 -127
  232. package/templates/welcome/src/index.deva +56 -8
  233. package/templates/welcome/src/variables.deva +2 -4
  234. package/tests/integration.rs +21 -0
  235. package/tests/rust/cli_check_build.rs +21 -0
  236. package/tests/rust/cli_help.rs +12 -0
  237. package/tests/rust/cli_template_list.rs +10 -0
  238. package/tests/rust/cli_version.rs +11 -0
  239. package/tests/typescript/index.spec.ts +136 -0
  240. package/tests/typescript/playhead.spec.ts +36 -0
  241. package/tests/typescript/render_e2e.spec.ts +77 -0
  242. package/tsconfig.json +12 -10
  243. package/typescript/bin/index.ts +19 -5
  244. package/typescript/core/functions/index.ts +83 -0
  245. package/typescript/core/index.ts +6 -0
  246. package/typescript/core/types/index.ts +4 -0
  247. package/typescript/core/types/plugin.ts +19 -0
  248. package/typescript/core/types/result.ts +29 -0
  249. package/typescript/core/types/statement.ts +47 -0
  250. package/typescript/core/types/value.ts +29 -0
  251. package/typescript/index.ts +8 -1
  252. package/typescript/pkg/devalang_core.d.ts +4 -0
  253. package/typescript/pkg/devalang_core.ts +49 -0
  254. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  255. package/typescript/scripts/postinstall.ts +85 -0
  256. package/typescript/scripts/version/bump.ts +0 -1
  257. package/typescript/scripts/version/fetch.ts +1 -6
  258. package/typescript/scripts/version/index.ts +0 -1
  259. package/docs/COMMANDS.md +0 -85
  260. package/docs/CONFIG.md +0 -30
  261. package/docs/SYNTAX.md +0 -210
  262. package/out-tsc/bin/devalang.exe +0 -0
  263. package/out-tsc/scripts/postbuild.js +0 -11
  264. package/rust/cli/build.rs +0 -137
  265. package/rust/cli/check.rs +0 -117
  266. package/rust/cli/play.rs +0 -193
  267. package/rust/config/loader.rs +0 -13
  268. package/rust/core/audio/engine.rs +0 -126
  269. package/rust/core/parser/handler/identifier.rs +0 -262
  270. package/rust/core/shared/duration.rs +0 -8
  271. package/rust/core/shared/mod.rs +0 -2
  272. package/rust/core/shared/value.rs +0 -18
  273. package/rust/core/utils/validation.rs +0 -35
  274. package/rust/utils/file.rs +0 -35
  275. package/rust/utils/signature.rs +0 -17
  276. package/rust/utils/version.rs +0 -15
  277. package/typescript/scripts/postbuild.ts +0 -8
@@ -1,123 +1,200 @@
1
- #[cfg(feature = "cli")]
2
- use crossterm::style::{Attribute, Color, ResetColor, SetAttribute, SetForegroundColor};
3
- use std::fmt::Write;
4
-
5
- #[derive(Debug, Clone, PartialEq)]
6
- pub enum LogLevel {
7
- Success,
8
- Error,
9
- Info,
10
- Warning,
11
- Watcher,
12
- Debug,
13
- }
14
-
15
- #[derive(Debug, Clone)]
16
- pub struct Logger;
17
-
18
- impl Logger {
19
- pub fn new() -> Self {
20
- Logger
21
- }
22
-
23
- // --- log_message ---
24
-
25
- #[cfg(feature = "cli")]
26
- pub fn log_message(&self, level: LogLevel, message: &str) {
27
- let formatted_status = self.format_status(level);
28
- println!("🦊 {} {} {}", self.language_signature(), formatted_status, message);
29
- }
30
-
31
- #[cfg(not(feature = "cli"))]
32
- pub fn log_message(&self, _level: LogLevel, _message: &str) {
33
- // no-op for WASM
34
- }
35
-
36
- // --- log_error_with_stacktrace ---
37
-
38
- #[cfg(feature = "cli")]
39
- pub fn log_error_with_stacktrace(&self, message: &str, stacktrace: &str) {
40
- let formatted_status = self.format_status(LogLevel::Error);
41
- println!("🦊 {} {} {}", self.language_signature(), formatted_status, message);
42
- println!(" ↳ {}", stacktrace);
43
- }
44
-
45
- #[cfg(not(feature = "cli"))]
46
- pub fn log_error_with_stacktrace(&self, _message: &str, _stacktrace: &str) {
47
- // no-op for WASM
48
- }
49
-
50
- // --- language_signature ---
51
-
52
- #[cfg(feature = "cli")]
53
- fn language_signature(&self) -> String {
54
- let mut s = String::new();
55
-
56
- write!(&mut s, "{}", SetForegroundColor(Color::Grey)).unwrap();
57
- s.push('[');
58
-
59
- write!(&mut s, "{}", SetForegroundColor(Color::Rgb { r: 29, g: 211, b: 176 })).unwrap();
60
- write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
61
- s.push_str("Devalang");
62
- write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
63
-
64
- write!(&mut s, "{}", SetForegroundColor(Color::Grey)).unwrap();
65
- s.push(']');
66
- write!(&mut s, "{}", ResetColor).unwrap();
67
-
68
- s
69
- }
70
-
71
- #[cfg(not(feature = "cli"))]
72
- fn language_signature(&self) -> String {
73
- "[Devalang]".to_string()
74
- }
75
-
76
- // --- format_status ---
77
-
78
- #[cfg(feature = "cli")]
79
- fn format_status(&self, level: LogLevel) -> String {
80
- let mut s = String::new();
81
-
82
- let color = match level {
83
- LogLevel::Success => Color::Rgb { r: 76, g: 175, b: 80 },
84
- LogLevel::Error => Color::Rgb { r: 244, g: 67, b: 54 },
85
- LogLevel::Info => Color::Rgb { r: 33, g: 150, b: 243 },
86
- LogLevel::Warning => Color::Rgb { r: 255, g: 152, b: 0 },
87
- LogLevel::Watcher => Color::Rgb { r: 156, g: 39, b: 176 },
88
- LogLevel::Debug => Color::Rgb { r: 103, g: 58, b: 183 },
89
- };
90
-
91
- let status = match level {
92
- LogLevel::Success => "SUCCESS",
93
- LogLevel::Error => "ERROR",
94
- LogLevel::Info => "INFO",
95
- LogLevel::Warning => "WARNING",
96
- LogLevel::Watcher => "WATCHER",
97
- LogLevel::Debug => "DEBUG",
98
- };
99
-
100
- s.push('[');
101
- write!(&mut s, "{}", SetForegroundColor(color)).unwrap();
102
- write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
103
- s.push_str(status);
104
- write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
105
- s.push(']');
106
- write!(&mut s, "{}", ResetColor).unwrap();
107
-
108
- s
109
- }
110
-
111
- #[cfg(not(feature = "cli"))]
112
- fn format_status(&self, level: LogLevel) -> String {
113
- match level {
114
- LogLevel::Success => "[SUCCESS]",
115
- LogLevel::Error => "[ERROR]",
116
- LogLevel::Info => "[INFO]",
117
- LogLevel::Warning => "[WARNING]",
118
- LogLevel::Watcher => "[WATCHER]",
119
- LogLevel::Debug => "[DEBUG]",
120
- }
121
- .to_string()
122
- }
123
- }
1
+ #[cfg(feature = "cli")]
2
+ use crossterm::style::{Attribute, Color, ResetColor, SetAttribute, SetForegroundColor};
3
+ #[cfg(feature = "cli")]
4
+ use std::fmt::Write;
5
+
6
+ #[derive(Debug, Clone, PartialEq)]
7
+ pub enum LogLevel {
8
+ Success,
9
+ Error,
10
+ Info,
11
+ Print,
12
+ Warning,
13
+ Watcher,
14
+ Debug,
15
+ }
16
+
17
+ #[derive(Debug, Clone)]
18
+ pub struct Logger;
19
+
20
+ impl Logger {
21
+ pub fn new() -> Self {
22
+ Logger
23
+ }
24
+
25
+ // Some methods are only used in CLI builds; silence dead_code warnings for non-cli.
26
+ #[cfg(not(feature = "cli"))]
27
+ #[allow(dead_code)]
28
+ fn __wasm_only_allow_dead_code(&self) {}
29
+
30
+ // Additionally allow dead_code for the CLI-only formatting helpers when building without cli
31
+ #[cfg(not(feature = "cli"))]
32
+ #[allow(dead_code)]
33
+ fn __wasm_only_format_helpers(&self) {}
34
+
35
+ #[cfg(feature = "cli")]
36
+ pub fn log_message(&self, level: LogLevel, message: &str) {
37
+ let formatted_status = self.format_status(level);
38
+ println!(
39
+ "🦊 {} {} {}",
40
+ self.language_signature(),
41
+ formatted_status,
42
+ message
43
+ );
44
+ }
45
+
46
+ #[cfg(not(feature = "cli"))]
47
+ pub fn log_message(&self, _level: LogLevel, _message: &str) {
48
+ // no-op for WASM
49
+ }
50
+
51
+ #[cfg(feature = "cli")]
52
+ pub fn log_message_with_trace(&self, level: LogLevel, message: &str, trace: Vec<&str>) {
53
+ let formatted_status = self.format_status(level);
54
+ println!(
55
+ "🦊 {} {} {}",
56
+ self.language_signature(),
57
+ formatted_status,
58
+ message
59
+ );
60
+ for t in trace {
61
+ println!(" ↳ {}", t);
62
+ }
63
+ }
64
+
65
+ #[cfg(not(feature = "cli"))]
66
+ pub fn log_message_with_trace(&self, _level: LogLevel, _message: &str, _trace: Vec<&str>) {
67
+ // no-op for WASM
68
+ }
69
+
70
+ #[cfg(feature = "cli")]
71
+ pub fn log_error_with_stacktrace(&self, message: &str, stacktrace: &str) {
72
+ let formatted_status = self.format_status(LogLevel::Error);
73
+ println!(
74
+ "🦊 {} {} {}",
75
+ self.language_signature(),
76
+ formatted_status,
77
+ message
78
+ );
79
+ println!(" ↳ {}", stacktrace);
80
+ }
81
+
82
+ #[cfg(not(feature = "cli"))]
83
+ pub fn log_error_with_stacktrace(&self, _message: &str, _stacktrace: &str) {
84
+ // no-op for WASM
85
+ }
86
+
87
+ #[cfg(feature = "cli")]
88
+ #[allow(dead_code)]
89
+ #[cfg_attr(not(feature = "cli"), allow(dead_code))]
90
+ fn language_signature(&self) -> String {
91
+ let mut s = String::new();
92
+
93
+ write!(&mut s, "{}", SetForegroundColor(Color::Grey)).unwrap();
94
+ s.push('[');
95
+
96
+ write!(
97
+ &mut s,
98
+ "{}",
99
+ SetForegroundColor(Color::Rgb {
100
+ r: 29,
101
+ g: 211,
102
+ b: 176,
103
+ })
104
+ )
105
+ .unwrap();
106
+ write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
107
+ s.push_str("Devalang");
108
+ write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
109
+
110
+ write!(&mut s, "{}", SetForegroundColor(Color::Grey)).unwrap();
111
+ s.push(']');
112
+ write!(&mut s, "{}", ResetColor).unwrap();
113
+
114
+ s
115
+ }
116
+
117
+ #[cfg(not(feature = "cli"))]
118
+ #[allow(dead_code)]
119
+ fn language_signature(&self) -> String {
120
+ "[Devalang]".to_string()
121
+ }
122
+
123
+ #[cfg(feature = "cli")]
124
+ fn format_status(&self, level: LogLevel) -> String {
125
+ let mut s = String::new();
126
+
127
+ let color = match level {
128
+ LogLevel::Success => Color::Rgb {
129
+ r: 76,
130
+ g: 175,
131
+ b: 80,
132
+ },
133
+ LogLevel::Error => Color::Rgb {
134
+ r: 244,
135
+ g: 67,
136
+ b: 54,
137
+ },
138
+ LogLevel::Info => Color::Rgb {
139
+ r: 33,
140
+ g: 150,
141
+ b: 243,
142
+ },
143
+ LogLevel::Warning => Color::Rgb {
144
+ r: 255,
145
+ g: 152,
146
+ b: 0,
147
+ },
148
+ LogLevel::Watcher => Color::Rgb {
149
+ r: 156,
150
+ g: 39,
151
+ b: 176,
152
+ },
153
+ LogLevel::Debug => Color::Rgb {
154
+ r: 103,
155
+ g: 58,
156
+ b: 183,
157
+ },
158
+ LogLevel::Print => Color::Rgb {
159
+ r: 255,
160
+ g: 255,
161
+ b: 255,
162
+ },
163
+ };
164
+
165
+ let status = match level {
166
+ LogLevel::Success => "SUCCESS",
167
+ LogLevel::Error => "ERROR",
168
+ LogLevel::Info => "INFO",
169
+ LogLevel::Warning => "WARNING",
170
+ LogLevel::Watcher => "WATCHER",
171
+ LogLevel::Debug => "DEBUG",
172
+ LogLevel::Print => "PRINT",
173
+ };
174
+
175
+ s.push('[');
176
+ write!(&mut s, "{}", SetForegroundColor(color)).unwrap();
177
+ write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
178
+ s.push_str(status);
179
+ write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
180
+ s.push(']');
181
+ write!(&mut s, "{}", ResetColor).unwrap();
182
+
183
+ s
184
+ }
185
+
186
+ #[cfg(not(feature = "cli"))]
187
+ #[allow(dead_code)]
188
+ fn format_status(&self, level: LogLevel) -> String {
189
+ (match level {
190
+ LogLevel::Success => "[SUCCESS]",
191
+ LogLevel::Error => "[ERROR]",
192
+ LogLevel::Info => "[INFO]",
193
+ LogLevel::Warning => "[WARNING]",
194
+ LogLevel::Watcher => "[WATCHER]",
195
+ LogLevel::Debug => "[DEBUG]",
196
+ LogLevel::Print => "[PRINT]",
197
+ })
198
+ .to_string()
199
+ }
200
+ }
@@ -0,0 +1,88 @@
1
+ use std::{
2
+ env, fs,
3
+ path::{Path, PathBuf},
4
+ };
5
+
6
+ pub const DEVALANG_CONFIG: &str = ".devalang";
7
+ pub const DEVA_DIR: &str = ".deva";
8
+
9
+ /// Returns the current working directory.
10
+ pub fn get_cwd() -> PathBuf {
11
+ // In wasm (and some restricted environments) `env::current_dir()` is unsupported
12
+ // and will return an error. Avoid panicking here and fall back to `.` so the
13
+ // runtime can still operate in a virtual environment.
14
+ env::current_dir().unwrap_or_else(|_| PathBuf::from("."))
15
+ }
16
+
17
+ /// Returns true if the given directory looks like a devalang project root.
18
+ /// Preference is given to the presence of `.devalang` (config file),
19
+ /// but falling back to a `.deva` directory is allowed.
20
+ pub fn is_project_root(dir: &Path) -> bool {
21
+ let config = dir.join(DEVALANG_CONFIG);
22
+ if config.is_file() {
23
+ return true;
24
+ }
25
+ let deva = dir.join(DEVA_DIR);
26
+ deva.is_dir()
27
+ }
28
+
29
+ /// Walks upward from `start` to locate the first directory considered a project root.
30
+ pub fn find_project_root_from(start: &Path) -> Option<PathBuf> {
31
+ for ancestor in start.ancestors() {
32
+ if is_project_root(ancestor) {
33
+ return Some(ancestor.to_path_buf());
34
+ }
35
+ }
36
+ None
37
+ }
38
+
39
+ /// Finds the project root from the current working directory.
40
+ pub fn find_project_root() -> Option<PathBuf> {
41
+ find_project_root_from(&get_cwd())
42
+ }
43
+
44
+ /// Finds the package root using the `CARGO_MANIFEST_DIR` env var set by Cargo.
45
+ pub fn get_package_root() -> Option<PathBuf> {
46
+ let cargo_dir = env::var("CARGO_MANIFEST_DIR").ok()?;
47
+ Some(PathBuf::from(cargo_dir))
48
+ }
49
+
50
+ /// Gets the project root or returns a descriptive error if not found.
51
+ pub fn get_project_root() -> Result<PathBuf, String> {
52
+ find_project_root()
53
+ .ok_or_else(|| "Project root not found. Run 'devalang init' in your project.".to_string())
54
+ }
55
+
56
+ /// Returns the path to `.devalang` in the project root, ensuring it exists.
57
+ pub fn get_devalang_config_path() -> Result<PathBuf, String> {
58
+ let root = get_project_root()?;
59
+ let config_path = root.join(DEVALANG_CONFIG);
60
+ if !config_path.exists() {
61
+ return Err(format!(
62
+ "Config file not found at '{}'. Please run 'devalang init' before continuing.",
63
+ config_path.display()
64
+ ));
65
+ }
66
+ Ok(config_path)
67
+ }
68
+
69
+ /// Returns the `.deva` directory inside the project root (without creating it).
70
+ pub fn get_deva_dir() -> Result<PathBuf, String> {
71
+ let root = get_project_root()?;
72
+ Ok(root.join(DEVA_DIR))
73
+ }
74
+
75
+ /// Ensures the `.deva` directory exists in the project root and returns its path.
76
+ pub fn ensure_deva_dir() -> Result<PathBuf, String> {
77
+ let deva = get_deva_dir()?;
78
+ if !deva.exists() {
79
+ fs::create_dir_all(&deva).map_err(|e| {
80
+ format!(
81
+ "Failed to create Deva directory '{}': {}",
82
+ deva.display(),
83
+ e
84
+ )
85
+ })?;
86
+ }
87
+ Ok(deva)
88
+ }
@@ -0,0 +1,41 @@
1
+ #[cfg(feature = "cli")]
2
+ pub fn get_signature(version: &str) -> String {
3
+ let signature = format!(
4
+ r#"
5
+ /|_/|
6
+ / ^ ^(_o 🦊 Devalang
7
+ / __.'
8
+ / \ A programming language for music and sound.
9
+ / _ \_ Part of the Devaloop project.
10
+ (_) (_) '._
11
+ '.__ '. .-''-'. https://devalang.com
12
+ ( '. ('.____.''
13
+ _) )'_, ) v{}
14
+ (__/ (__/
15
+ "#,
16
+ version
17
+ );
18
+
19
+ signature
20
+ }
21
+
22
+ #[cfg(not(feature = "cli"))]
23
+ pub fn get_signature(version: &str) -> String {
24
+ let signature = format!(
25
+ r#"
26
+ /|_/|
27
+ / ^ ^(_o 🦊 Devalang
28
+ / __.'
29
+ / \ A programming language for music and sound.
30
+ / _ \_ Part of the Devaloop project.
31
+ (_) (_) '._
32
+ '.__ '. .-''-'. https://devalang.com
33
+ ( '. ('.____.''
34
+ _) )'_, ) v{}
35
+ (__/ (__/
36
+ "#,
37
+ version
38
+ );
39
+
40
+ signature
41
+ }
@@ -1,21 +1,20 @@
1
- #[cfg(feature = "cli")]
2
- use indicatif::{ ProgressBar, ProgressStyle };
3
- use std::{ time::Duration };
4
-
5
- #[cfg(feature = "cli")]
6
- pub fn with_spinner<T, F>(start_msg: &str, f: F) -> T where F: FnOnce() -> T {
7
- let spinner = ProgressBar::new_spinner();
8
- spinner.set_style(
9
- ProgressStyle::with_template("{spinner:.green} {msg}")
10
- .unwrap()
11
- .tick_strings(&["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"])
12
- );
13
- spinner.set_message(start_msg.to_string());
14
- spinner.enable_steady_tick(Duration::from_millis(80));
15
-
16
- let result = f();
17
-
18
- spinner.finish_and_clear();
19
-
20
- result
21
- }
1
+ #[cfg(feature = "cli")]
2
+ use indicatif::{ProgressBar, ProgressStyle};
3
+ #[cfg(feature = "cli")]
4
+ use std::time::Duration;
5
+
6
+ #[cfg(feature = "cli")]
7
+ pub fn start_spinner(start_msg: &str) -> ProgressBar {
8
+ let spinner = ProgressBar::new_spinner();
9
+ spinner.set_style(
10
+ ProgressStyle::with_template("{spinner:.green} {msg}")
11
+ .unwrap()
12
+ .tick_strings(&["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"]),
13
+ );
14
+ spinner.set_message(start_msg.to_string());
15
+ spinner.enable_steady_tick(Duration::from_millis(80));
16
+
17
+ std::thread::sleep(Duration::from_millis(750));
18
+
19
+ spinner
20
+ }
@@ -0,0 +1,27 @@
1
+ use crate::{path::get_package_root, signature::get_signature};
2
+
3
+ pub fn get_version() -> String {
4
+ if let Some(root) = get_package_root() {
5
+ let project_version_json = root.join("project-version.json");
6
+ if let Ok(version) = std::fs::read_to_string(project_version_json) {
7
+ if let Ok(parsed) = serde_json::from_str::<serde_json::Value>(&version) {
8
+ if let Some(version_val) = parsed.get("version") {
9
+ if let Some(s) = version_val.as_str() {
10
+ return s.to_string();
11
+ }
12
+ }
13
+ }
14
+ }
15
+ }
16
+
17
+ "0.0.0".to_string()
18
+ }
19
+
20
+ pub fn get_version_with_signature() -> String {
21
+ let version = get_version();
22
+ let signature = get_signature(&version);
23
+
24
+ println!("{}", signature);
25
+
26
+ "(c) 2025 Devaloop. All rights reserved.".to_string()
27
+ }
@@ -1,33 +1,46 @@
1
- use notify::{ Watcher, RecursiveMode, Config, RecommendedWatcher };
2
- use std::sync::mpsc::channel;
3
-
4
- use std::time::{ Duration, Instant };
5
-
6
- pub fn watch_directory<F>(entry: String, callback: F) -> notify::Result<()>
7
- where F: Fn() + Send + 'static
8
- {
9
- let (tx, rx) = channel();
10
-
11
- let mut watcher: RecommendedWatcher = Watcher::new(tx, Config::default())?;
12
- watcher.watch(&entry.as_ref(), RecursiveMode::Recursive)?;
13
-
14
- let mut last_trigger = Instant::now();
15
-
16
- loop {
17
- match rx.recv() {
18
- Ok(_) => {
19
- let now = Instant::now();
20
- if now.duration_since(last_trigger) > Duration::from_millis(200) {
21
- callback();
22
- last_trigger = now;
23
- }
24
- }
25
- Err(e) => {
26
- eprintln!("Channel error: {:?}", e);
27
- break;
28
- }
29
- }
30
- }
31
-
32
- Ok(())
33
- }
1
+ use notify::{Config, RecommendedWatcher, RecursiveMode, Watcher};
2
+ use std::sync::mpsc::channel;
3
+
4
+ use std::time::{Duration, Instant};
5
+
6
+ pub fn watch_directory<F>(entry: String, callback: F) -> notify::Result<()>
7
+ where
8
+ F: Fn() + Send + 'static,
9
+ {
10
+ let (tx, rx) = channel();
11
+
12
+ let mut watcher: RecommendedWatcher = Watcher::new(tx, Config::default())?;
13
+ watcher.watch(&entry.as_ref(), RecursiveMode::Recursive)?;
14
+
15
+ let mut last_trigger = Instant::now();
16
+
17
+ loop {
18
+ match rx.recv() {
19
+ Ok(_) => {
20
+ let now = Instant::now();
21
+ if now.duration_since(last_trigger) > Duration::from_millis(200) {
22
+ callback();
23
+ last_trigger = now;
24
+ }
25
+ }
26
+ Err(e) => {
27
+ // Prefer structured logging when available
28
+ #[cfg(feature = "cli")]
29
+ {
30
+ let logger = crate::logger::Logger::new();
31
+ logger.log_message(
32
+ crate::logger::LogLevel::Error,
33
+ &format!("Channel error: {:?}", e),
34
+ );
35
+ }
36
+ #[cfg(not(feature = "cli"))]
37
+ {
38
+ eprintln!("Channel error: {:?}", e);
39
+ }
40
+ break;
41
+ }
42
+ }
43
+ }
44
+
45
+ Ok(())
46
+ }
@@ -0,0 +1,5 @@
1
+ pub fn get_api_url() -> String {
2
+ let api_url = "https://api.devalang.com";
3
+ // let api_url = "http://127.0.0.1:8989";
4
+ api_url.to_string()
5
+ }
@@ -0,0 +1,34 @@
1
+ use std::error::Error;
2
+ use std::fs::File;
3
+ use std::io::{Cursor, copy};
4
+ use std::path::Path;
5
+
6
+ pub fn get_cdn_url() -> String {
7
+ let cdn_url = "https://cdn.devalang.com";
8
+ // let cdn_url = "http://127.0.0.1:8888";
9
+ cdn_url.to_string()
10
+ }
11
+
12
+ pub async fn download_from_cdn(url: &str, destination: &Path) -> Result<(), Box<dyn Error>> {
13
+ if !url.starts_with(&get_cdn_url()) {
14
+ return Err(format!("Invalid CDN URL: {}", url).into());
15
+ }
16
+
17
+ let response = reqwest::get(url).await?;
18
+
19
+ if !response.status().is_success() {
20
+ return Err(format!("Failed to download file: HTTP {}", response.status()).into());
21
+ }
22
+
23
+ if let Some(parent) = destination.parent() {
24
+ std::fs::create_dir_all(parent)?;
25
+ }
26
+
27
+ let bytes = response.bytes().await?;
28
+ let mut content = Cursor::new(bytes);
29
+ let mut file = File::create(destination)?;
30
+
31
+ copy(&mut content, &mut file)?;
32
+
33
+ Ok(())
34
+ }
@@ -0,0 +1,3 @@
1
+ pub mod api;
2
+ pub mod cdn;
3
+ pub mod sso;
@@ -0,0 +1,5 @@
1
+ pub fn get_sso_url() -> String {
2
+ let sso_url = "https://sso.devalang.com";
3
+ // let sso_url = "http://localhost:5174";
4
+ sso_url.to_string()
5
+ }