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

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (322) hide show
  1. package/.cargo/config.toml +2 -0
  2. package/.devalang +6 -1
  3. package/.github/workflows/ci.yml +103 -0
  4. package/Cargo.toml +81 -48
  5. package/README.md +137 -154
  6. package/docs/CHANGELOG.md +428 -1
  7. package/docs/CONTRIBUTING.md +101 -0
  8. package/docs/ROADMAP.md +14 -7
  9. package/docs/TODO.md +16 -15
  10. package/examples/automation.deva +42 -0
  11. package/examples/bank.deva +7 -0
  12. package/examples/bus.deva +10 -0
  13. package/examples/duration.deva +9 -0
  14. package/examples/effect.deva +2 -0
  15. package/examples/events.deva +12 -0
  16. package/examples/filter.deva +11 -0
  17. package/examples/function.deva +15 -0
  18. package/examples/index.deva +57 -12
  19. package/examples/lfo.deva +9 -0
  20. package/examples/loop.deva +5 -12
  21. package/examples/pattern.deva +8 -0
  22. package/examples/plugin.deva +16 -0
  23. package/examples/synth.deva +11 -1
  24. package/examples/synth_types.deva +17 -0
  25. package/examples/variables.deva +1 -1
  26. package/out-tsc/bin/index.d.ts +2 -0
  27. package/out-tsc/bin/index.js +51 -7
  28. package/out-tsc/core/functions/index.d.ts +42 -0
  29. package/out-tsc/core/functions/index.js +87 -0
  30. package/out-tsc/core/index.d.ts +6 -0
  31. package/out-tsc/core/index.js +22 -0
  32. package/out-tsc/core/types/index.d.ts +4 -0
  33. package/out-tsc/core/types/index.js +20 -0
  34. package/out-tsc/core/types/plugin.d.ts +18 -0
  35. package/out-tsc/core/types/plugin.js +2 -0
  36. package/out-tsc/core/types/result.d.ts +27 -0
  37. package/out-tsc/core/types/result.js +2 -0
  38. package/out-tsc/core/types/statement.d.ts +106 -0
  39. package/out-tsc/core/types/statement.js +2 -0
  40. package/out-tsc/core/types/value.d.ts +43 -0
  41. package/out-tsc/core/types/value.js +2 -0
  42. package/out-tsc/index.d.ts +7 -0
  43. package/out-tsc/index.js +42 -1
  44. package/out-tsc/pkg/devalang_core.d.ts +15 -0
  45. package/out-tsc/pkg/devalang_core.js +65 -0
  46. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +34 -0
  47. package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
  48. package/out-tsc/scripts/copy-wasm-dts.js +73 -0
  49. package/out-tsc/scripts/postinstall.d.ts +1 -0
  50. package/out-tsc/scripts/postinstall.js +83 -0
  51. package/out-tsc/scripts/version/bump.d.ts +1 -0
  52. package/out-tsc/scripts/version/fetch.d.ts +1 -0
  53. package/out-tsc/scripts/version/index.d.ts +1 -0
  54. package/out-tsc/scripts/version/sync.d.ts +1 -0
  55. package/package.json +28 -7
  56. package/project-version.json +4 -4
  57. package/rust/cli/bank/api.rs +122 -0
  58. package/rust/cli/bank/commands.rs +306 -0
  59. package/rust/cli/bank/mod.rs +29 -0
  60. package/rust/cli/build/commands.rs +153 -0
  61. package/rust/cli/build/mod.rs +2 -0
  62. package/rust/cli/build/process.rs +165 -0
  63. package/rust/cli/check/mod.rs +208 -0
  64. package/rust/cli/discover/commands.rs +253 -0
  65. package/rust/cli/discover/config.rs +111 -0
  66. package/rust/cli/discover/fs.rs +19 -0
  67. package/rust/cli/discover/install.rs +103 -0
  68. package/rust/cli/discover/metadata.rs +48 -0
  69. package/rust/cli/discover/mod.rs +5 -0
  70. package/rust/cli/{init.rs → init/commands.rs} +32 -23
  71. package/rust/cli/init/mod.rs +1 -0
  72. package/rust/cli/install/addon.rs +118 -0
  73. package/rust/cli/install/bank.rs +72 -0
  74. package/rust/cli/install/commands.rs +35 -0
  75. package/rust/cli/install/mod.rs +4 -0
  76. package/rust/cli/install/plugin.rs +80 -0
  77. package/rust/cli/login/commands.rs +124 -0
  78. package/rust/cli/login/mod.rs +1 -0
  79. package/rust/cli/mod.rs +9 -202
  80. package/rust/cli/parser.rs +359 -0
  81. package/rust/cli/play/commands.rs +375 -0
  82. package/rust/cli/play/io.rs +17 -0
  83. package/rust/cli/play/mod.rs +5 -0
  84. package/rust/cli/play/process.rs +159 -0
  85. package/rust/cli/play/realtime.rs +91 -0
  86. package/rust/cli/play/utils.rs +23 -0
  87. package/rust/cli/telemetry/commands.rs +22 -0
  88. package/rust/cli/telemetry/event_creator.rs +80 -0
  89. package/rust/cli/telemetry/mod.rs +3 -0
  90. package/rust/cli/telemetry/send.rs +51 -0
  91. package/rust/cli/{template.rs → template/commands.rs} +17 -5
  92. package/rust/cli/template/mod.rs +1 -0
  93. package/rust/cli/update/commands.rs +6 -0
  94. package/rust/cli/update/mod.rs +1 -0
  95. package/rust/config/driver.rs +112 -0
  96. package/rust/config/mod.rs +3 -16
  97. package/rust/config/ops.rs +26 -0
  98. package/rust/config/settings.rs +101 -0
  99. package/rust/core/audio/engine/driver.rs +220 -0
  100. package/rust/core/audio/engine/export.rs +169 -0
  101. package/rust/core/audio/engine/helpers.rs +178 -0
  102. package/rust/core/audio/engine/mod.rs +56 -0
  103. package/rust/core/audio/engine/notes/dsp.rs +85 -0
  104. package/rust/core/audio/engine/notes/mod.rs +44 -0
  105. package/rust/core/audio/engine/notes/params.rs +294 -0
  106. package/rust/core/audio/engine/sample/insert.rs +199 -0
  107. package/rust/core/audio/engine/sample/mod.rs +40 -0
  108. package/rust/core/audio/engine/sample/padding.rs +170 -0
  109. package/rust/core/audio/evaluator/condition.rs +61 -0
  110. package/rust/core/audio/evaluator/mod.rs +9 -0
  111. package/rust/core/audio/evaluator/numeric.rs +152 -0
  112. package/rust/core/audio/evaluator/rhs.rs +16 -0
  113. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  114. package/rust/core/audio/interpreter/driver.rs +574 -216
  115. package/rust/core/audio/interpreter/mod.rs +2 -12
  116. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
  117. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
  118. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
  119. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
  120. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  121. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  122. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  123. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  124. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  125. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  126. package/rust/core/audio/interpreter/statements/automate.rs +16 -0
  127. package/rust/core/audio/interpreter/statements/call.rs +295 -0
  128. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -69
  129. package/rust/core/audio/interpreter/statements/function.rs +24 -0
  130. package/rust/core/audio/interpreter/statements/let_.rs +36 -0
  131. package/rust/core/audio/interpreter/statements/load.rs +17 -0
  132. package/rust/core/audio/interpreter/statements/loop_.rs +115 -0
  133. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  134. package/rust/core/audio/interpreter/statements/sleep.rs +28 -0
  135. package/rust/core/audio/interpreter/statements/spawn.rs +253 -0
  136. package/rust/core/audio/interpreter/statements/tempo.rs +40 -0
  137. package/rust/core/audio/interpreter/statements/trigger.rs +239 -0
  138. package/rust/core/audio/loader/mod.rs +1 -1
  139. package/rust/core/audio/loader/trigger.rs +98 -52
  140. package/rust/core/audio/mod.rs +2 -2
  141. package/rust/core/audio/player.rs +28 -12
  142. package/rust/core/audio/special/easing.rs +189 -0
  143. package/rust/core/audio/special/env.rs +45 -0
  144. package/rust/core/audio/special/math.rs +134 -0
  145. package/rust/core/audio/special/mod.rs +9 -0
  146. package/rust/core/audio/special/modulator.rs +143 -0
  147. package/rust/core/builder/mod.rs +129 -80
  148. package/rust/core/debugger/lexer.rs +4 -4
  149. package/rust/core/debugger/logs.rs +52 -0
  150. package/rust/core/debugger/mod.rs +11 -2
  151. package/rust/core/debugger/preprocessor.rs +4 -4
  152. package/rust/core/debugger/store.rs +38 -25
  153. package/rust/core/error/mod.rs +221 -12
  154. package/rust/core/lexer/driver.rs +59 -0
  155. package/rust/core/lexer/handler/arrow.rs +62 -11
  156. package/rust/core/lexer/handler/at.rs +5 -5
  157. package/rust/core/lexer/handler/brace.rs +11 -11
  158. package/rust/core/lexer/handler/colon.rs +5 -5
  159. package/rust/core/lexer/handler/comment.rs +3 -3
  160. package/rust/core/lexer/handler/dot.rs +6 -6
  161. package/rust/core/lexer/handler/driver.rs +143 -32
  162. package/rust/core/lexer/handler/identifier.rs +11 -5
  163. package/rust/core/lexer/handler/indent.rs +18 -4
  164. package/rust/core/lexer/handler/mod.rs +6 -5
  165. package/rust/core/lexer/handler/newline.rs +3 -3
  166. package/rust/core/lexer/handler/number.rs +5 -5
  167. package/rust/core/lexer/handler/operator.rs +5 -3
  168. package/rust/core/lexer/handler/parenthesis.rs +41 -0
  169. package/rust/core/lexer/handler/slash.rs +21 -0
  170. package/rust/core/lexer/handler/string.rs +3 -3
  171. package/rust/core/lexer/mod.rs +1 -49
  172. package/rust/core/lexer/token.rs +17 -12
  173. package/rust/core/mod.rs +9 -10
  174. package/rust/core/parser/driver/block.rs +111 -0
  175. package/rust/core/parser/driver/cursor.rs +82 -0
  176. package/rust/core/parser/driver/driver_impl.rs +139 -0
  177. package/rust/core/parser/driver/mod.rs +6 -0
  178. package/rust/core/parser/driver/parse_array.rs +120 -0
  179. package/rust/core/parser/driver/parse_map.rs +223 -0
  180. package/rust/core/parser/driver/parser.rs +160 -0
  181. package/rust/core/parser/handler/arrow_call.rs +277 -126
  182. package/rust/core/parser/handler/at.rs +142 -25
  183. package/rust/core/parser/handler/bank.rs +83 -20
  184. package/rust/core/parser/handler/condition.rs +14 -5
  185. package/rust/core/parser/handler/dot.rs +111 -75
  186. package/rust/core/parser/handler/identifier/automate.rs +254 -0
  187. package/rust/core/parser/handler/identifier/call.rs +74 -24
  188. package/rust/core/parser/handler/identifier/emit.rs +70 -0
  189. package/rust/core/parser/handler/identifier/function.rs +113 -0
  190. package/rust/core/parser/handler/identifier/group.rs +28 -14
  191. package/rust/core/parser/handler/identifier/let_.rs +61 -21
  192. package/rust/core/parser/handler/identifier/mod.rs +24 -20
  193. package/rust/core/parser/handler/identifier/on.rs +107 -0
  194. package/rust/core/parser/handler/identifier/print.rs +49 -0
  195. package/rust/core/parser/handler/identifier/sleep.rs +77 -14
  196. package/rust/core/parser/handler/identifier/spawn.rs +81 -31
  197. package/rust/core/parser/handler/identifier/synth.rs +102 -32
  198. package/rust/core/parser/handler/loop_.rs +144 -22
  199. package/rust/core/parser/handler/mod.rs +6 -5
  200. package/rust/core/parser/handler/pattern.rs +74 -0
  201. package/rust/core/parser/handler/tempo.rs +67 -9
  202. package/rust/core/parser/mod.rs +3 -4
  203. package/rust/core/parser/statement.rs +6 -92
  204. package/rust/core/plugin/loader.rs +137 -0
  205. package/rust/core/plugin/mod.rs +2 -0
  206. package/rust/core/plugin/runner/mod.rs +11 -0
  207. package/rust/core/plugin/runner/non_wasm.rs +297 -0
  208. package/rust/core/plugin/runner/wasm32.rs +43 -0
  209. package/rust/core/preprocessor/loader/inject.rs +278 -0
  210. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  211. package/rust/core/preprocessor/loader/mod.rs +235 -0
  212. package/rust/core/preprocessor/mod.rs +4 -4
  213. package/rust/core/preprocessor/module.rs +55 -50
  214. package/rust/core/preprocessor/processor/handlers.rs +107 -0
  215. package/rust/core/preprocessor/processor/mod.rs +1 -0
  216. package/rust/core/preprocessor/resolver/bank.rs +14 -12
  217. package/rust/core/preprocessor/resolver/call.rs +106 -105
  218. package/rust/core/preprocessor/resolver/condition.rs +13 -10
  219. package/rust/core/preprocessor/resolver/driver.rs +145 -48
  220. package/rust/core/preprocessor/resolver/function.rs +69 -0
  221. package/rust/core/preprocessor/resolver/group.rs +122 -61
  222. package/rust/core/preprocessor/resolver/let_.rs +13 -12
  223. package/rust/core/preprocessor/resolver/loop_.rs +240 -13
  224. package/rust/core/preprocessor/resolver/mod.rs +8 -6
  225. package/rust/core/preprocessor/resolver/pattern.rs +83 -0
  226. package/rust/core/preprocessor/resolver/spawn.rs +83 -42
  227. package/rust/core/preprocessor/resolver/synth.rs +15 -11
  228. package/rust/core/preprocessor/resolver/tempo.rs +13 -14
  229. package/rust/core/preprocessor/resolver/trigger.rs +32 -28
  230. package/rust/core/preprocessor/resolver/value.rs +111 -13
  231. package/rust/core/store/global.rs +57 -39
  232. package/rust/core/store/mod.rs +0 -3
  233. package/rust/lib.rs +323 -117
  234. package/rust/main.rs +388 -65
  235. package/rust/types/Cargo.toml +11 -0
  236. package/rust/types/src/addons.rs +55 -0
  237. package/rust/types/src/ast.rs +202 -0
  238. package/rust/types/src/config.rs +84 -0
  239. package/rust/types/src/lib.rs +15 -0
  240. package/rust/types/src/plugin.rs +20 -0
  241. package/rust/types/src/store.rs +139 -0
  242. package/rust/types/src/telemetry.rs +85 -0
  243. package/rust/utils/Cargo.toml +26 -0
  244. package/rust/utils/src/error.rs +186 -0
  245. package/rust/utils/src/file.rs +94 -0
  246. package/rust/utils/src/first_usage.rs +97 -0
  247. package/rust/utils/{mod.rs → src/lib.rs} +6 -3
  248. package/rust/utils/{logger.rs → src/logger.rs} +94 -17
  249. package/rust/utils/src/path.rs +129 -0
  250. package/rust/utils/src/signature.rs +41 -0
  251. package/rust/utils/{spinner.rs → src/spinner.rs} +7 -8
  252. package/rust/utils/src/version.rs +27 -0
  253. package/rust/utils/{watcher.rs → src/watcher.rs} +17 -4
  254. package/rust/web/api.rs +5 -0
  255. package/rust/web/cdn.rs +34 -0
  256. package/rust/web/mod.rs +3 -0
  257. package/rust/web/sso.rs +5 -0
  258. package/templates/minimal/README.md +143 -127
  259. package/templates/welcome/README.md +143 -127
  260. package/templates/welcome/src/index.deva +56 -8
  261. package/templates/welcome/src/variables.deva +2 -4
  262. package/tests/integration.rs +21 -0
  263. package/tests/rust/cli_check_build.rs +21 -0
  264. package/tests/rust/cli_help.rs +12 -0
  265. package/tests/rust/cli_template_list.rs +10 -0
  266. package/tests/rust/cli_version.rs +11 -0
  267. package/tests/typescript/index.spec.ts +136 -0
  268. package/tests/typescript/playhead.spec.ts +36 -0
  269. package/tests/typescript/render_e2e.spec.ts +77 -0
  270. package/tsconfig.json +12 -10
  271. package/typescript/bin/index.ts +19 -5
  272. package/typescript/core/functions/index.ts +94 -0
  273. package/typescript/core/index.ts +6 -0
  274. package/typescript/core/types/index.ts +4 -0
  275. package/typescript/core/types/plugin.ts +19 -0
  276. package/typescript/core/types/result.ts +29 -0
  277. package/typescript/core/types/statement.ts +47 -0
  278. package/typescript/core/types/value.ts +29 -0
  279. package/typescript/index.ts +8 -1
  280. package/typescript/pkg/devalang_core.d.ts +4 -0
  281. package/typescript/pkg/devalang_core.ts +65 -0
  282. package/typescript/scripts/copy-wasm-dts.ts +41 -0
  283. package/typescript/scripts/postinstall.ts +85 -0
  284. package/typescript/scripts/version/bump.ts +0 -1
  285. package/typescript/scripts/version/index.ts +0 -1
  286. package/docs/COMMANDS.md +0 -85
  287. package/docs/CONFIG.md +0 -30
  288. package/docs/SYNTAX.md +0 -210
  289. package/out-tsc/bin/devalang.exe +0 -0
  290. package/out-tsc/scripts/postbuild.js +0 -11
  291. package/rust/cli/build.rs +0 -137
  292. package/rust/cli/check.rs +0 -117
  293. package/rust/cli/play.rs +0 -193
  294. package/rust/config/loader.rs +0 -13
  295. package/rust/core/audio/engine.rs +0 -203
  296. package/rust/core/audio/evaluator.rs +0 -31
  297. package/rust/core/audio/interpreter/arrow_call.rs +0 -129
  298. package/rust/core/audio/interpreter/call.rs +0 -64
  299. package/rust/core/audio/interpreter/let_.rs +0 -19
  300. package/rust/core/audio/interpreter/load.rs +0 -18
  301. package/rust/core/audio/interpreter/loop_.rs +0 -67
  302. package/rust/core/audio/interpreter/sleep.rs +0 -36
  303. package/rust/core/audio/interpreter/spawn.rs +0 -66
  304. package/rust/core/audio/interpreter/tempo.rs +0 -16
  305. package/rust/core/audio/interpreter/trigger.rs +0 -69
  306. package/rust/core/audio/renderer.rs +0 -54
  307. package/rust/core/parser/driver.rs +0 -331
  308. package/rust/core/preprocessor/loader.rs +0 -193
  309. package/rust/core/preprocessor/processor.rs +0 -76
  310. package/rust/core/shared/duration.rs +0 -8
  311. package/rust/core/shared/mod.rs +0 -2
  312. package/rust/core/shared/value.rs +0 -18
  313. package/rust/core/store/export.rs +0 -28
  314. package/rust/core/store/import.rs +0 -28
  315. package/rust/core/store/variable.rs +0 -28
  316. package/rust/core/utils/mod.rs +0 -2
  317. package/rust/core/utils/path.rs +0 -31
  318. package/rust/core/utils/validation.rs +0 -37
  319. package/rust/utils/file.rs +0 -35
  320. package/rust/utils/signature.rs +0 -17
  321. package/rust/utils/version.rs +0 -15
  322. package/typescript/scripts/postbuild.ts +0 -8
@@ -0,0 +1,129 @@
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
+ }
89
+
90
+ /// Finds the entry file given a path, returning the normalized path if found.
91
+ /// If the path is a directory, it looks for `index.deva` inside it.
92
+ /// Returns None if no valid entry file is found.
93
+ pub fn find_entry_file(entry: &str) -> Option<String> {
94
+ let path = Path::new(entry);
95
+
96
+ if path.is_file() {
97
+ return Some(normalize_path(entry));
98
+ }
99
+
100
+ if path.is_dir() {
101
+ let candidate = path.join("index.deva");
102
+ if candidate.exists() {
103
+ return Some(normalize_path(&candidate));
104
+ }
105
+ }
106
+
107
+ None
108
+ }
109
+
110
+ /// Normalizes a path to use forward slashes and removes redundant components.
111
+ pub fn normalize_path<P: AsRef<Path>>(path: P) -> String {
112
+ let path_buf = PathBuf::from(path.as_ref());
113
+ path_buf
114
+ .components()
115
+ .collect::<PathBuf>()
116
+ .to_string_lossy()
117
+ .replace('\\', "/")
118
+ }
119
+
120
+ /// Resolves a relative import path against a base path, normalizing the result.
121
+ pub fn resolve_relative_path(base: &str, import: &str) -> String {
122
+ let base_path = Path::new(base).parent().unwrap_or_else(|| Path::new(""));
123
+ let full_path = base_path.join(import);
124
+ full_path
125
+ .components()
126
+ .collect::<PathBuf>()
127
+ .to_string_lossy()
128
+ .replace("\\", "/")
129
+ }
@@ -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
1
  #[cfg(feature = "cli")]
2
- use indicatif::{ ProgressBar, ProgressStyle };
3
- use std::{ time::Duration };
2
+ use indicatif::{ProgressBar, ProgressStyle};
3
+ #[cfg(feature = "cli")]
4
+ use std::time::Duration;
4
5
 
5
6
  #[cfg(feature = "cli")]
6
- pub fn with_spinner<T, F>(start_msg: &str, f: F) -> T where F: FnOnce() -> T {
7
+ pub fn start_spinner(start_msg: &str) -> ProgressBar {
7
8
  let spinner = ProgressBar::new_spinner();
8
9
  spinner.set_style(
9
10
  ProgressStyle::with_template("{spinner:.green} {msg}")
10
11
  .unwrap()
11
- .tick_strings(&["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"])
12
+ .tick_strings(&["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"]),
12
13
  );
13
14
  spinner.set_message(start_msg.to_string());
14
15
  spinner.enable_steady_tick(Duration::from_millis(80));
15
16
 
16
- let result = f();
17
-
18
- spinner.finish_and_clear();
17
+ std::thread::sleep(Duration::from_millis(750));
19
18
 
20
- result
19
+ spinner
21
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("package.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,10 +1,11 @@
1
- use notify::{ Watcher, RecursiveMode, Config, RecommendedWatcher };
1
+ use notify::{Config, RecommendedWatcher, RecursiveMode, Watcher};
2
2
  use std::sync::mpsc::channel;
3
3
 
4
- use std::time::{ Duration, Instant };
4
+ use std::time::{Duration, Instant};
5
5
 
6
6
  pub fn watch_directory<F>(entry: String, callback: F) -> notify::Result<()>
7
- where F: Fn() + Send + 'static
7
+ where
8
+ F: Fn() + Send + 'static,
8
9
  {
9
10
  let (tx, rx) = channel();
10
11
 
@@ -23,7 +24,19 @@ pub fn watch_directory<F>(entry: String, callback: F) -> notify::Result<()>
23
24
  }
24
25
  }
25
26
  Err(e) => {
26
- eprintln!("Channel error: {:?}", 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
+ }
27
40
  break;
28
41
  }
29
42
  }
@@ -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
+ }
@@ -1,192 +1,204 @@
1
1
  <div align="center">
2
- <img src="https://firebasestorage.googleapis.com/v0/b/devaloop-labs.firebasestorage.app/o/devalang-teal-logo.svg?alt=media&token=d2a5705a-1eba-4b49-88e6-895a761fb7f7" alt="Devalang Logo">
2
+ <img src="https://devalang.com/images/devalang-logo-min.png" alt="Devalang Logo" width="100" />
3
3
  </div>
4
4
 
5
5
  ![Rust](https://img.shields.io/badge/Made%20with-Rust-orange?logo=rust)
6
6
  ![TypeScript](https://img.shields.io/badge/Built%20with-TypeScript-blue?logo=typescript)
7
7
  ![Node.js](https://img.shields.io/badge/Node.js-18%2B-brightgreen?logo=node.js)
8
8
 
9
- ![Project Status](https://img.shields.io/badge/status-alpha-red)
10
- ![Version](https://img.shields.io/badge/version-0.0.1-blue)
9
+ ![Project Status](https://img.shields.io/badge/status-beta-blue)
10
+ ![Version](https://img.shields.io/npm/v/@devaloop/devalang)
11
11
  ![License: MIT](https://img.shields.io/badge/license-MIT-green)
12
- ![Platform](https://img.shields.io/badge/platform-Windows-blue)
12
+
13
+ ![Linux](https://img.shields.io/badge/linux-supported-blue?logo=linux)
14
+ ![macOS](https://img.shields.io/badge/macOS-supported-blue?logo=apple)
15
+ ![Windows](https://img.shields.io/badge/windows-supported-blue?logo=windows)
13
16
 
14
17
  ![npm](https://img.shields.io/npm/dt/@devaloop/devalang)
15
18
  ![crates](https://img.shields.io/crates/d/devalang)
16
19
 
17
- ## 🎼 Devalang, by **Devaloop Labs**
20
+ ![GitHub Actions Workflow Status](https://img.shields.io/github/actions/workflow/status/devaloop-labs/devalang/.github/workflows/ci.yml)
18
21
 
19
- 🎶 Compose music with code — simple, structured, sonic.
22
+ # 🦊 Devalang (CORE) — Compose music with code
20
23
 
21
24
  Devalang is a tiny domain-specific language (DSL) for music makers, sound designers, and audio hackers.
22
25
  Compose loops, control samples, render and play audio — all in clean, readable text.
23
26
 
24
- 🦊 Whether you're building a track, shaping textures, or performing live, Devalang helps you think in rhythms. It’s designed to be simple, expressive, and fast — because your ideas shouldn’t wait.
27
+ Whether you're building a track, shaping textures, or performing live, Devalang helps you think in rhythms. It’s designed to be simple, expressive, and fast — because your ideas shouldn’t wait.
25
28
 
26
29
  From studio sketches to live sets, Devalang gives you rhythmic control — with the elegance of code.
27
30
 
28
- > 🚧 **v0.0.1-alpha.5 Notice** 🚧
31
+ > **🚧 Notice 🚧**
29
32
  >
30
- > Currently, Devalang CLI is only available for **Windows**.
31
- > Linux and macOS binaries will be added in future releases via cross-platform builds.
32
-
33
- ---
33
+ > Includes synthesis, playback, and rendering features, but is still in early development, and breaking changes may occur.
34
+ >
35
+ > **NEW**: [Devaforge is now available for creating addons](https://github.com/devaloop-labs/devaforge).
36
+ >
37
+ > **NEW**: Now available for Windows, Linux, and macOS.
34
38
 
35
39
  ## 📚 Quick Access
36
40
 
37
- - [📖 Documentation](./docs/)
41
+ - [▶️ Playground](https://playground.devalang.com)
42
+ - [📖 Documentation](https://docs.devalang.com)
43
+ - [🧩 VSCode Extension](https://marketplace.visualstudio.com/items?itemName=devaloop.devalang-vscode)
44
+ - [🎨 Prettier Plugin](https://www.npmjs.com/package/@devaloop/prettier-plugin-devalang)
45
+ - [📜 Changelog](./docs/CHANGELOG.md)
38
46
  - [💡 Examples](./examples/)
39
47
  - [🌐 Project Website](https://devalang.com)
48
+ - [📦 Devaforge on npm](https://www.npmjs.com/package/@devaloop/devaforge)
49
+ - [📦 Devalang on npm](https://www.npmjs.com/package/@devaloop/devalang)
40
50
 
41
- ## 🚀 Features
42
-
43
- - 🎵 **Audio Engine**: Integrated audio playback and rendering
44
- - 🧩 **Module system** for importing and exporting variables between files
45
- - 📜 **Structured AST** generation for debugging and future compilation
46
- - 🔢 **Basic data types**: strings, numbers, booleans, maps, arrays
47
- - 👁️ **Watch mode** for `build`, `check` and `play` commands
48
- - 📂 **Project templates** for quick setup
51
+ ## ⏱️ Try it now !
49
52
 
50
- ## 📆 Installation
53
+ ### Try Devalang in your browser
51
54
 
52
- ### For users
55
+ > [Have a look at the Playground to try Devalang directly in your browser](https://playground.devalang.com)
53
56
 
54
- > - ⚠️ Requires [Node.js 18+](https://nodejs.org/en/download)
57
+ ### Try Devalang in your terminal
55
58
 
56
- Install the package globally (NPM)
59
+ #### With Node.js
57
60
 
58
61
  ```bash
59
- npm install -g @devaloop/devalang
62
+ npm install -g @devaloop/devalang@latest
60
63
  ```
61
64
 
62
- Usage without install (NPX)
65
+ #### With Rust
63
66
 
64
67
  ```bash
65
- npx @devaloop/devalang <command>
68
+ cargo install devalang
66
69
  ```
67
70
 
68
- ### For contributors
69
-
70
- > - ⚠️ Requires [Node.js 18+](https://nodejs.org/en/download)
71
- > - ⚠️ Requires [Rust 1.70+](https://www.rust-lang.org/learn/get-started#installing-rust)
71
+ #### Initialize a new project
72
72
 
73
73
  ```bash
74
- > git clone https://github.com/devaloop-labs/devalang.git
75
- > cd devalang
76
- > npm install
77
- > cargo install --path .
74
+ devalang init --name my-project --template minimal
78
75
  ```
79
76
 
80
- Development usage (you can customize arguments in package.json)
81
-
82
- ```bash
83
- # For syntax checking test
84
- npm run rust:dev:check
85
- # For building test
86
- npm run rust:dev:build
87
- ```
88
-
89
- ## ❔ Usage
90
-
91
- NOTE: Commands are available via `devalang` or `npx @devaloop/devalang`.
92
-
93
- NOTE: Arguments can be passed to commands using `--<argument>` syntax. You can also use a configuration file to set default values for various settings, making it easier to manage your Devalang project.
77
+ #### Write your first script
94
78
 
95
- NOTE: Some commands require a mandatory `--entry` argument to specify the input folder, and a `--output` argument to specify the output folder. If not specified, they default to `./src` and `./output` respectively.
79
+ Create a new Devalang file `src/index.deva` in the project directory:
96
80
 
97
- For more examples, see [docs/COMMANDS.md](./docs/COMMANDS.md)
98
-
99
- ### Initialize a new project
100
-
101
- In the current directory
102
-
103
- ```bash
104
- devalang init
105
- ```
106
-
107
- Or use optional arguments to specify a directory name and a template
108
-
109
- ```bash
110
- devalang init --name <project-name> --template <template-name>
111
- ```
112
-
113
- ### Checking syntax only
114
-
115
- ```bash
116
- devalang check --watch
81
+ ```deva
82
+ # src/index.deva
83
+
84
+ # BPM definition
85
+ bpm 125
86
+
87
+ # Bank picking (make sure you've installed it)
88
+ bank devaloop.808 as my808Bank
89
+
90
+ # Pattern literal without options
91
+ pattern kickPattern with my808Bank.kick = "x--- x--- x--- x---"
92
+
93
+ group myGroup:
94
+ # Rhythmic (each beat playing a kick)
95
+ # on beat:
96
+ # .my808Bank.kick 1/4
97
+
98
+ # Synth definition with ADSR
99
+ let myLead = synth sine {
100
+ attack: 0,
101
+ decay: 100,
102
+ sustain: 100,
103
+ release: 100
104
+ }
105
+
106
+ # Global automation
107
+ automate myLead:
108
+ param volume {
109
+ 0% = 0.0
110
+ 100% = 0.5
111
+ }
112
+ param pitch {
113
+ 0% = -12.0
114
+ 100% = 12.0
115
+ }
116
+
117
+ # Notes in a loop with condition
118
+ for i in [1, 2, 3]:
119
+ if i == 3:
120
+ myLead -> note(C5, { duration: 200 })
121
+ print "Playing note C5 for " + i
122
+
123
+ # Pause runtime for 500ms
124
+ sleep 500
125
+
126
+ # Note with automation
127
+ myLead -> note(C4, {
128
+ duration: 400,
129
+ velocity: 0.8,
130
+ automate: {
131
+ pan: {
132
+ 0%: -1.0,
133
+ 100%: 0.0
134
+ }
135
+ }
136
+ })
137
+
138
+ # Notes with params
139
+ myLead -> note(G4, { duration: 600, glide: true, target_freq: 659.25 })
140
+ myLead -> note(B3, { duration: 400, slide: true, target_amp: 0.3 })
141
+
142
+ # Spawning the group & the pattern to play them in parallel
143
+ spawn myGroup
144
+ spawn kickPattern
117
145
  ```
118
146
 
119
- ### Building output files
120
-
121
- ```bash
122
- devalang build --watch
123
- ```
147
+ ### And the best part ? You can play it directly from the command line:
124
148
 
125
- ### Playing audio files (once by file change)
149
+ #### Play the script once
126
150
 
127
151
  ```bash
128
- devalang play --watch
152
+ devalang play
129
153
  ```
130
154
 
131
- ### Playing audio files (continuous playback, even without file changes)
155
+ #### **LIVE mode** (repeat the playback + watch mode)
132
156
 
133
157
  ```bash
134
158
  devalang play --repeat
135
159
  ```
136
160
 
137
- ## ⚙️ Configuration
161
+ ### 🎉 You can now hear your Devalang code in action
138
162
 
139
- You can use a configuration file to set default values for various settings, making it easier to manage your Devalang project.
163
+ > For more examples, check out the [examples directory](./examples/)
140
164
 
141
- To do this, create a `.devalang` file in the root of your project directory.
165
+ ## Why Devalang ?
142
166
 
143
- See [docs/CONFIG.md](./docs/CONFIG.md) for more information.
167
+ - 🎹 Prototype audio ideas without opening a DAW, even VSCode with our Playground
168
+ - 💻 Integrate sound into code-based workflows
169
+ - 🎛️ Control audio parameters through readable syntax
170
+ - 🧪 Build musical logic with variables and conditions
171
+ - 🔄 Create complex patterns with ease
144
172
 
145
- ## 📄 Syntax example
146
-
147
- For more examples, see [docs/SYNTAX.md](./docs/SYNTAX.md)
148
-
149
- ```deva
150
- # index.deva
151
-
152
- @import { globalBpm, globalBank, kickDuration } from "global.deva"
153
-
154
- @load "./examples/samples/kick-808.wav" as customKick
155
-
156
- bpm globalBpm
157
- # Will declare the tempo at the globalBpm variable beats per minute
158
-
159
- bank globalBank
160
- # Will declare a custom instrument bank using the globalBank variable
161
-
162
- loop 5:
163
- .customKick kickDuration {reverb=50, drive=25}
164
- # Will play 5 times a kick for the duration of the kickDuration variable with reverb and drive effects
165
- ```
166
-
167
- ```deva
168
- # global.deva
169
-
170
- let globalBpm = 120
171
- let globalBank = 808
172
- let kickDuration = 500
173
-
174
- @export { globalBpm, globalBank, kickDuration }
175
- ```
176
-
177
- ## 🧯 Known issues
178
-
179
- - No support yet for `if`, `else`, `else if` statements
180
- - No support yet for `@group`, `@pattern`, `@function` statements
181
- - No support yet for cross-platform builds (Linux, macOS)
182
-
183
- ## 🧪 Roadmap Highlights
184
-
185
- For more info, see [docs/ROADMAP.md](./docs/ROADMAP.md)
173
+ ## 🚀 Features
186
174
 
187
- - Other statements (e.g `if`, `@group`, ...)
188
- - Cross-platform support (Linux, macOS)
189
- - More built-in instruments (e.g. snare, hi-hat, etc.)
175
+ - **Fast Build & Hot Reload** — optimized build process for quicker iteration.
176
+ - 🎵 **Audio Engine & Real-time runner** — low-latency playback, render-to-file, and a realtime runner used by `devalang play` for live feedback.
177
+ - ▶️ **Live mode (watch + repeat)** — edit and hear changes instantly with `devalang play --repeat` and watch mode.
178
+ - 🧩 **Language primitives** — synths, notes, ADSR, maps, arrays, loops, conditionals and functions for expressive musical logic.
179
+ - 🎛️ **Per-note automation & modulators** — `automate` maps, `$mod.*`, `$easing.*` and `$math.*` helpers for envelopes and LFOs.
180
+ - 🧩 **Module system & structured AST** — import/export variables, stable AST output for debugging and tooling.
181
+ - 🧰 **Plugins & Addons (WASM-ready)** — install plugins/banks, `@use` directive, and WASM plugin integration so plugins can render or process audio at runtime.
182
+ - 📦 **Addon manager & Devaforge** — CLI commands to discover/install banks, plugins and templates; `devaforge` helps create addons.
183
+ - ⚙️ **CLI tooling** — `build`, `check`, `play`, `install`, `init`, `discover`, `telemetry` and more with consistent flags (`--watch`, `--debug`, `--compress`).
184
+ - 📂 **Project templates & examples** — quick-start templates and many example projects in `examples/`.
185
+ - 🧑‍💻 **TypeScript API & WASM distribution** — Node-friendly package with TypeScript bindings and a WASM build for browser/Node usage.
186
+ - 🧰 **Editor & formatting support** — VSCode extension and Prettier plugin to edit Devalang with syntax and formatting support.
187
+ - 🎵 **Custom samples & banks** — drop samples into `.deva` and reference them from code; banks of sounds for fast composition.
188
+ - 🔄 **Looping, grouping & scheduling** — precise beat-tied scheduling primitives for complex rhythmic patterns.
189
+
190
+ ## 📄 Documentation
191
+
192
+ ### [Please refer to the online documentation](https://docs.devalang.com) for detailed information on syntax, features, and usage examples
193
+
194
+ ## 📰 What's new
195
+
196
+ - **Devaforge**: Introduced a new system for creating and managing addons, including a CLI for addon generation.
197
+ - **Documentation updates**: Improved documentation for clarity and completeness.
198
+ - **Discovering addons**: Introduced a new command to detect addons.
199
+ - **Public TypeScript API**: Added a public TypeScript API for easier integration.
200
+ - **Improved error messages**: Enhanced error messages for better debugging.
201
+ - **Bug fixes**: Various bug fixes and stability improvements.
190
202
 
191
203
  ## 🛡️ License
192
204
 
@@ -197,6 +209,10 @@ MIT — see [LICENSE](./LICENSE)
197
209
  Contributions, bug reports and suggestions are welcome !
198
210
  Feel free to open an issue or submit a pull request.
199
211
 
212
+ For more info, see [docs/CONTRIBUTING.md](./docs/CONTRIBUTING.md).
213
+
200
214
  ## 📢 Contact
201
215
 
216
+ Feel free to reach out for any inquiries or feedback.
217
+
202
218
  📧 [contact@devaloop.com](mailto:contact@devaloop.com)