@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.
- package/.cargo/config.toml +2 -0
- package/.devalang +6 -1
- package/.github/workflows/ci.yml +103 -0
- package/Cargo.toml +81 -48
- package/README.md +137 -154
- package/docs/CHANGELOG.md +428 -1
- package/docs/CONTRIBUTING.md +101 -0
- package/docs/ROADMAP.md +14 -7
- package/docs/TODO.md +16 -15
- package/examples/automation.deva +42 -0
- package/examples/bank.deva +7 -0
- package/examples/bus.deva +10 -0
- package/examples/duration.deva +9 -0
- package/examples/effect.deva +2 -0
- package/examples/events.deva +12 -0
- package/examples/filter.deva +11 -0
- package/examples/function.deva +15 -0
- package/examples/index.deva +57 -12
- package/examples/lfo.deva +9 -0
- package/examples/loop.deva +5 -12
- package/examples/pattern.deva +8 -0
- package/examples/plugin.deva +16 -0
- package/examples/synth.deva +11 -1
- package/examples/synth_types.deva +17 -0
- package/examples/variables.deva +1 -1
- package/out-tsc/bin/index.d.ts +2 -0
- package/out-tsc/bin/index.js +51 -7
- package/out-tsc/core/functions/index.d.ts +42 -0
- package/out-tsc/core/functions/index.js +87 -0
- package/out-tsc/core/index.d.ts +6 -0
- package/out-tsc/core/index.js +22 -0
- package/out-tsc/core/types/index.d.ts +4 -0
- package/out-tsc/core/types/index.js +20 -0
- package/out-tsc/core/types/plugin.d.ts +18 -0
- package/out-tsc/core/types/plugin.js +2 -0
- package/out-tsc/core/types/result.d.ts +27 -0
- package/out-tsc/core/types/result.js +2 -0
- package/out-tsc/core/types/statement.d.ts +106 -0
- package/out-tsc/core/types/statement.js +2 -0
- package/out-tsc/core/types/value.d.ts +43 -0
- package/out-tsc/core/types/value.js +2 -0
- package/out-tsc/index.d.ts +7 -0
- package/out-tsc/index.js +42 -1
- package/out-tsc/pkg/devalang_core.d.ts +15 -0
- package/out-tsc/pkg/devalang_core.js +65 -0
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +34 -0
- package/out-tsc/scripts/copy-wasm-dts.d.ts +1 -0
- package/out-tsc/scripts/copy-wasm-dts.js +73 -0
- package/out-tsc/scripts/postinstall.d.ts +1 -0
- package/out-tsc/scripts/postinstall.js +83 -0
- package/out-tsc/scripts/version/bump.d.ts +1 -0
- package/out-tsc/scripts/version/fetch.d.ts +1 -0
- package/out-tsc/scripts/version/index.d.ts +1 -0
- package/out-tsc/scripts/version/sync.d.ts +1 -0
- package/package.json +28 -7
- package/project-version.json +4 -4
- package/rust/cli/bank/api.rs +122 -0
- package/rust/cli/bank/commands.rs +306 -0
- package/rust/cli/bank/mod.rs +29 -0
- package/rust/cli/build/commands.rs +153 -0
- package/rust/cli/build/mod.rs +2 -0
- package/rust/cli/build/process.rs +165 -0
- package/rust/cli/check/mod.rs +208 -0
- package/rust/cli/discover/commands.rs +253 -0
- package/rust/cli/discover/config.rs +111 -0
- package/rust/cli/discover/fs.rs +19 -0
- package/rust/cli/discover/install.rs +103 -0
- package/rust/cli/discover/metadata.rs +48 -0
- package/rust/cli/discover/mod.rs +5 -0
- package/rust/cli/{init.rs → init/commands.rs} +32 -23
- package/rust/cli/init/mod.rs +1 -0
- package/rust/cli/install/addon.rs +118 -0
- package/rust/cli/install/bank.rs +72 -0
- package/rust/cli/install/commands.rs +35 -0
- package/rust/cli/install/mod.rs +4 -0
- package/rust/cli/install/plugin.rs +80 -0
- package/rust/cli/login/commands.rs +124 -0
- package/rust/cli/login/mod.rs +1 -0
- package/rust/cli/mod.rs +9 -202
- package/rust/cli/parser.rs +359 -0
- package/rust/cli/play/commands.rs +375 -0
- package/rust/cli/play/io.rs +17 -0
- package/rust/cli/play/mod.rs +5 -0
- package/rust/cli/play/process.rs +159 -0
- package/rust/cli/play/realtime.rs +91 -0
- package/rust/cli/play/utils.rs +23 -0
- package/rust/cli/telemetry/commands.rs +22 -0
- package/rust/cli/telemetry/event_creator.rs +80 -0
- package/rust/cli/telemetry/mod.rs +3 -0
- package/rust/cli/telemetry/send.rs +51 -0
- package/rust/cli/{template.rs → template/commands.rs} +17 -5
- package/rust/cli/template/mod.rs +1 -0
- package/rust/cli/update/commands.rs +6 -0
- package/rust/cli/update/mod.rs +1 -0
- package/rust/config/driver.rs +112 -0
- package/rust/config/mod.rs +3 -16
- package/rust/config/ops.rs +26 -0
- package/rust/config/settings.rs +101 -0
- package/rust/core/audio/engine/driver.rs +220 -0
- package/rust/core/audio/engine/export.rs +169 -0
- package/rust/core/audio/engine/helpers.rs +178 -0
- package/rust/core/audio/engine/mod.rs +56 -0
- package/rust/core/audio/engine/notes/dsp.rs +85 -0
- package/rust/core/audio/engine/notes/mod.rs +44 -0
- package/rust/core/audio/engine/notes/params.rs +294 -0
- package/rust/core/audio/engine/sample/insert.rs +199 -0
- package/rust/core/audio/engine/sample/mod.rs +40 -0
- package/rust/core/audio/engine/sample/padding.rs +170 -0
- package/rust/core/audio/evaluator/condition.rs +61 -0
- package/rust/core/audio/evaluator/mod.rs +9 -0
- package/rust/core/audio/evaluator/numeric.rs +152 -0
- package/rust/core/audio/evaluator/rhs.rs +16 -0
- package/rust/core/audio/evaluator/string_expr.rs +94 -0
- package/rust/core/audio/interpreter/driver.rs +574 -216
- package/rust/core/audio/interpreter/mod.rs +2 -12
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +175 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +384 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +2 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +316 -0
- package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
- package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
- package/rust/core/audio/interpreter/statements/automate.rs +16 -0
- package/rust/core/audio/interpreter/statements/call.rs +295 -0
- package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -69
- package/rust/core/audio/interpreter/statements/function.rs +24 -0
- package/rust/core/audio/interpreter/statements/let_.rs +36 -0
- package/rust/core/audio/interpreter/statements/load.rs +17 -0
- package/rust/core/audio/interpreter/statements/loop_.rs +115 -0
- package/rust/core/audio/interpreter/statements/mod.rs +12 -0
- package/rust/core/audio/interpreter/statements/sleep.rs +28 -0
- package/rust/core/audio/interpreter/statements/spawn.rs +253 -0
- package/rust/core/audio/interpreter/statements/tempo.rs +40 -0
- package/rust/core/audio/interpreter/statements/trigger.rs +239 -0
- package/rust/core/audio/loader/mod.rs +1 -1
- package/rust/core/audio/loader/trigger.rs +98 -52
- package/rust/core/audio/mod.rs +2 -2
- package/rust/core/audio/player.rs +28 -12
- package/rust/core/audio/special/easing.rs +189 -0
- package/rust/core/audio/special/env.rs +45 -0
- package/rust/core/audio/special/math.rs +134 -0
- package/rust/core/audio/special/mod.rs +9 -0
- package/rust/core/audio/special/modulator.rs +143 -0
- package/rust/core/builder/mod.rs +129 -80
- package/rust/core/debugger/lexer.rs +4 -4
- package/rust/core/debugger/logs.rs +52 -0
- package/rust/core/debugger/mod.rs +11 -2
- package/rust/core/debugger/preprocessor.rs +4 -4
- package/rust/core/debugger/store.rs +38 -25
- package/rust/core/error/mod.rs +221 -12
- package/rust/core/lexer/driver.rs +59 -0
- package/rust/core/lexer/handler/arrow.rs +62 -11
- package/rust/core/lexer/handler/at.rs +5 -5
- package/rust/core/lexer/handler/brace.rs +11 -11
- package/rust/core/lexer/handler/colon.rs +5 -5
- package/rust/core/lexer/handler/comment.rs +3 -3
- package/rust/core/lexer/handler/dot.rs +6 -6
- package/rust/core/lexer/handler/driver.rs +143 -32
- package/rust/core/lexer/handler/identifier.rs +11 -5
- package/rust/core/lexer/handler/indent.rs +18 -4
- package/rust/core/lexer/handler/mod.rs +6 -5
- package/rust/core/lexer/handler/newline.rs +3 -3
- package/rust/core/lexer/handler/number.rs +5 -5
- package/rust/core/lexer/handler/operator.rs +5 -3
- package/rust/core/lexer/handler/parenthesis.rs +41 -0
- package/rust/core/lexer/handler/slash.rs +21 -0
- package/rust/core/lexer/handler/string.rs +3 -3
- package/rust/core/lexer/mod.rs +1 -49
- package/rust/core/lexer/token.rs +17 -12
- package/rust/core/mod.rs +9 -10
- package/rust/core/parser/driver/block.rs +111 -0
- package/rust/core/parser/driver/cursor.rs +82 -0
- package/rust/core/parser/driver/driver_impl.rs +139 -0
- package/rust/core/parser/driver/mod.rs +6 -0
- package/rust/core/parser/driver/parse_array.rs +120 -0
- package/rust/core/parser/driver/parse_map.rs +223 -0
- package/rust/core/parser/driver/parser.rs +160 -0
- package/rust/core/parser/handler/arrow_call.rs +277 -126
- package/rust/core/parser/handler/at.rs +142 -25
- package/rust/core/parser/handler/bank.rs +83 -20
- package/rust/core/parser/handler/condition.rs +14 -5
- package/rust/core/parser/handler/dot.rs +111 -75
- package/rust/core/parser/handler/identifier/automate.rs +254 -0
- package/rust/core/parser/handler/identifier/call.rs +74 -24
- package/rust/core/parser/handler/identifier/emit.rs +70 -0
- package/rust/core/parser/handler/identifier/function.rs +113 -0
- package/rust/core/parser/handler/identifier/group.rs +28 -14
- package/rust/core/parser/handler/identifier/let_.rs +61 -21
- package/rust/core/parser/handler/identifier/mod.rs +24 -20
- package/rust/core/parser/handler/identifier/on.rs +107 -0
- package/rust/core/parser/handler/identifier/print.rs +49 -0
- package/rust/core/parser/handler/identifier/sleep.rs +77 -14
- package/rust/core/parser/handler/identifier/spawn.rs +81 -31
- package/rust/core/parser/handler/identifier/synth.rs +102 -32
- package/rust/core/parser/handler/loop_.rs +144 -22
- package/rust/core/parser/handler/mod.rs +6 -5
- package/rust/core/parser/handler/pattern.rs +74 -0
- package/rust/core/parser/handler/tempo.rs +67 -9
- package/rust/core/parser/mod.rs +3 -4
- package/rust/core/parser/statement.rs +6 -92
- package/rust/core/plugin/loader.rs +137 -0
- package/rust/core/plugin/mod.rs +2 -0
- package/rust/core/plugin/runner/mod.rs +11 -0
- package/rust/core/plugin/runner/non_wasm.rs +297 -0
- package/rust/core/plugin/runner/wasm32.rs +43 -0
- package/rust/core/preprocessor/loader/inject.rs +278 -0
- package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
- package/rust/core/preprocessor/loader/mod.rs +235 -0
- package/rust/core/preprocessor/mod.rs +4 -4
- package/rust/core/preprocessor/module.rs +55 -50
- package/rust/core/preprocessor/processor/handlers.rs +107 -0
- package/rust/core/preprocessor/processor/mod.rs +1 -0
- package/rust/core/preprocessor/resolver/bank.rs +14 -12
- package/rust/core/preprocessor/resolver/call.rs +106 -105
- package/rust/core/preprocessor/resolver/condition.rs +13 -10
- package/rust/core/preprocessor/resolver/driver.rs +145 -48
- package/rust/core/preprocessor/resolver/function.rs +69 -0
- package/rust/core/preprocessor/resolver/group.rs +122 -61
- package/rust/core/preprocessor/resolver/let_.rs +13 -12
- package/rust/core/preprocessor/resolver/loop_.rs +240 -13
- package/rust/core/preprocessor/resolver/mod.rs +8 -6
- package/rust/core/preprocessor/resolver/pattern.rs +83 -0
- package/rust/core/preprocessor/resolver/spawn.rs +83 -42
- package/rust/core/preprocessor/resolver/synth.rs +15 -11
- package/rust/core/preprocessor/resolver/tempo.rs +13 -14
- package/rust/core/preprocessor/resolver/trigger.rs +32 -28
- package/rust/core/preprocessor/resolver/value.rs +111 -13
- package/rust/core/store/global.rs +57 -39
- package/rust/core/store/mod.rs +0 -3
- package/rust/lib.rs +323 -117
- package/rust/main.rs +388 -65
- package/rust/types/Cargo.toml +11 -0
- package/rust/types/src/addons.rs +55 -0
- package/rust/types/src/ast.rs +202 -0
- package/rust/types/src/config.rs +84 -0
- package/rust/types/src/lib.rs +15 -0
- package/rust/types/src/plugin.rs +20 -0
- package/rust/types/src/store.rs +139 -0
- package/rust/types/src/telemetry.rs +85 -0
- package/rust/utils/Cargo.toml +26 -0
- package/rust/utils/src/error.rs +186 -0
- package/rust/utils/src/file.rs +94 -0
- package/rust/utils/src/first_usage.rs +97 -0
- package/rust/utils/{mod.rs → src/lib.rs} +6 -3
- package/rust/utils/{logger.rs → src/logger.rs} +94 -17
- package/rust/utils/src/path.rs +129 -0
- package/rust/utils/src/signature.rs +41 -0
- package/rust/utils/{spinner.rs → src/spinner.rs} +7 -8
- package/rust/utils/src/version.rs +27 -0
- package/rust/utils/{watcher.rs → src/watcher.rs} +17 -4
- package/rust/web/api.rs +5 -0
- package/rust/web/cdn.rs +34 -0
- package/rust/web/mod.rs +3 -0
- package/rust/web/sso.rs +5 -0
- package/templates/minimal/README.md +143 -127
- package/templates/welcome/README.md +143 -127
- package/templates/welcome/src/index.deva +56 -8
- package/templates/welcome/src/variables.deva +2 -4
- package/tests/integration.rs +21 -0
- package/tests/rust/cli_check_build.rs +21 -0
- package/tests/rust/cli_help.rs +12 -0
- package/tests/rust/cli_template_list.rs +10 -0
- package/tests/rust/cli_version.rs +11 -0
- package/tests/typescript/index.spec.ts +136 -0
- package/tests/typescript/playhead.spec.ts +36 -0
- package/tests/typescript/render_e2e.spec.ts +77 -0
- package/tsconfig.json +12 -10
- package/typescript/bin/index.ts +19 -5
- package/typescript/core/functions/index.ts +94 -0
- package/typescript/core/index.ts +6 -0
- package/typescript/core/types/index.ts +4 -0
- package/typescript/core/types/plugin.ts +19 -0
- package/typescript/core/types/result.ts +29 -0
- package/typescript/core/types/statement.ts +47 -0
- package/typescript/core/types/value.ts +29 -0
- package/typescript/index.ts +8 -1
- package/typescript/pkg/devalang_core.d.ts +4 -0
- package/typescript/pkg/devalang_core.ts +65 -0
- package/typescript/scripts/copy-wasm-dts.ts +41 -0
- package/typescript/scripts/postinstall.ts +85 -0
- package/typescript/scripts/version/bump.ts +0 -1
- package/typescript/scripts/version/index.ts +0 -1
- package/docs/COMMANDS.md +0 -85
- package/docs/CONFIG.md +0 -30
- package/docs/SYNTAX.md +0 -210
- package/out-tsc/bin/devalang.exe +0 -0
- package/out-tsc/scripts/postbuild.js +0 -11
- package/rust/cli/build.rs +0 -137
- package/rust/cli/check.rs +0 -117
- package/rust/cli/play.rs +0 -193
- package/rust/config/loader.rs +0 -13
- package/rust/core/audio/engine.rs +0 -203
- package/rust/core/audio/evaluator.rs +0 -31
- package/rust/core/audio/interpreter/arrow_call.rs +0 -129
- package/rust/core/audio/interpreter/call.rs +0 -64
- package/rust/core/audio/interpreter/let_.rs +0 -19
- package/rust/core/audio/interpreter/load.rs +0 -18
- package/rust/core/audio/interpreter/loop_.rs +0 -67
- package/rust/core/audio/interpreter/sleep.rs +0 -36
- package/rust/core/audio/interpreter/spawn.rs +0 -66
- package/rust/core/audio/interpreter/tempo.rs +0 -16
- package/rust/core/audio/interpreter/trigger.rs +0 -69
- package/rust/core/audio/renderer.rs +0 -54
- package/rust/core/parser/driver.rs +0 -331
- package/rust/core/preprocessor/loader.rs +0 -193
- package/rust/core/preprocessor/processor.rs +0 -76
- package/rust/core/shared/duration.rs +0 -8
- package/rust/core/shared/mod.rs +0 -2
- package/rust/core/shared/value.rs +0 -18
- package/rust/core/store/export.rs +0 -28
- package/rust/core/store/import.rs +0 -28
- package/rust/core/store/variable.rs +0 -28
- package/rust/core/utils/mod.rs +0 -2
- package/rust/core/utils/path.rs +0 -31
- package/rust/core/utils/validation.rs +0 -37
- package/rust/utils/file.rs +0 -35
- package/rust/utils/signature.rs +0 -17
- package/rust/utils/version.rs +0 -15
- package/typescript/scripts/postbuild.ts +0 -8
|
@@ -1,192 +1,204 @@
|
|
|
1
1
|
<div align="center">
|
|
2
|
-
<img src="https://
|
|
2
|
+
<img src="https://devalang.com/images/devalang-logo-min.png" alt="Devalang Logo" width="100" />
|
|
3
3
|
</div>
|
|
4
4
|
|
|
5
5
|

|
|
6
6
|

|
|
7
7
|

|
|
8
8
|
|
|
9
|
-

|
|
10
|
+

|
|
11
11
|

|
|
12
|
-
|
|
12
|
+
|
|
13
|
+

|
|
14
|
+

|
|
15
|
+

|
|
13
16
|
|
|
14
17
|

|
|
15
18
|

|
|
16
19
|
|
|
17
|
-
|
|
20
|
+

|
|
18
21
|
|
|
19
|
-
|
|
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
|
-
|
|
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
|
-
>
|
|
31
|
+
> **🚧 Notice 🚧**
|
|
29
32
|
>
|
|
30
|
-
>
|
|
31
|
-
>
|
|
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
|
-
- [
|
|
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
|
-
##
|
|
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
|
-
|
|
53
|
+
### Try Devalang in your browser
|
|
51
54
|
|
|
52
|
-
|
|
55
|
+
> [Have a look at the Playground to try Devalang directly in your browser](https://playground.devalang.com)
|
|
53
56
|
|
|
54
|
-
|
|
57
|
+
### Try Devalang in your terminal
|
|
55
58
|
|
|
56
|
-
|
|
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
|
-
|
|
65
|
+
#### With Rust
|
|
63
66
|
|
|
64
67
|
```bash
|
|
65
|
-
|
|
68
|
+
cargo install devalang
|
|
66
69
|
```
|
|
67
70
|
|
|
68
|
-
|
|
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
|
-
|
|
75
|
-
> cd devalang
|
|
76
|
-
> npm install
|
|
77
|
-
> cargo install --path .
|
|
74
|
+
devalang init --name my-project --template minimal
|
|
78
75
|
```
|
|
79
76
|
|
|
80
|
-
|
|
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
|
-
|
|
79
|
+
Create a new Devalang file `src/index.deva` in the project directory:
|
|
96
80
|
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
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
|
-
###
|
|
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
|
-
|
|
149
|
+
#### Play the script once
|
|
126
150
|
|
|
127
151
|
```bash
|
|
128
|
-
devalang play
|
|
152
|
+
devalang play
|
|
129
153
|
```
|
|
130
154
|
|
|
131
|
-
|
|
155
|
+
#### **LIVE mode** (repeat the playback + watch mode)
|
|
132
156
|
|
|
133
157
|
```bash
|
|
134
158
|
devalang play --repeat
|
|
135
159
|
```
|
|
136
160
|
|
|
137
|
-
|
|
161
|
+
### 🎉 You can now hear your Devalang code in action
|
|
138
162
|
|
|
139
|
-
|
|
163
|
+
> For more examples, check out the [examples directory](./examples/)
|
|
140
164
|
|
|
141
|
-
|
|
165
|
+
## ❓ Why Devalang ?
|
|
142
166
|
|
|
143
|
-
|
|
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
|
-
##
|
|
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
|
-
-
|
|
188
|
-
-
|
|
189
|
-
-
|
|
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)
|
|
@@ -1,13 +1,61 @@
|
|
|
1
|
-
# 🦊 Welcome to Devalang
|
|
2
|
-
#
|
|
3
|
-
# Let's start with a simple example that plays a sample in a loop.
|
|
1
|
+
# 🦊 Welcome to Devalang
|
|
2
|
+
# Happy live-coding !
|
|
4
3
|
|
|
5
|
-
@import { bpmVar
|
|
6
|
-
@load "../samples/kick-808.wav" as
|
|
4
|
+
@import { bpmVar } from './variables.deva'
|
|
5
|
+
@load "../samples/kick-808.wav" as myKick
|
|
7
6
|
|
|
7
|
+
# BPM definition
|
|
8
8
|
bpm bpmVar
|
|
9
9
|
|
|
10
|
-
|
|
10
|
+
group myGroup:
|
|
11
|
+
# Rhythmic (each beat playing a kick)
|
|
12
|
+
on beat:
|
|
13
|
+
.myKick 1/4
|
|
11
14
|
|
|
12
|
-
|
|
13
|
-
|
|
15
|
+
# Synth definition with ADSR
|
|
16
|
+
let myLead = synth sine {
|
|
17
|
+
attack: 0,
|
|
18
|
+
decay: 100,
|
|
19
|
+
sustain: 100,
|
|
20
|
+
release: 100
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
# Global automation
|
|
24
|
+
automate myLead:
|
|
25
|
+
param volume {
|
|
26
|
+
0% = 0.0
|
|
27
|
+
100% = 0.5
|
|
28
|
+
}
|
|
29
|
+
param pitch {
|
|
30
|
+
0% = -12.0
|
|
31
|
+
100% = 12.0
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
# Notes in a loop with condition
|
|
35
|
+
for i in [1, 2, 3]:
|
|
36
|
+
if i == 3:
|
|
37
|
+
myLead -> note(C5, { duration: 200 })
|
|
38
|
+
print "Playing note C5 for " + i
|
|
39
|
+
|
|
40
|
+
# Pause runtime for 500ms
|
|
41
|
+
sleep 500
|
|
42
|
+
|
|
43
|
+
# Note with automation
|
|
44
|
+
myLead -> note(C4, {
|
|
45
|
+
duration: 400,
|
|
46
|
+
velocity: 0.8,
|
|
47
|
+
automate: {
|
|
48
|
+
pan: {
|
|
49
|
+
0%: -1.0,
|
|
50
|
+
100%: 0.0
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
})
|
|
54
|
+
|
|
55
|
+
# Notes with params
|
|
56
|
+
myLead -> note(E4, { duration: 400 })
|
|
57
|
+
myLead -> note(G4, { duration: 600, glide: true, target_freq: 659.25 })
|
|
58
|
+
myLead -> note(B3, { duration: 400, slide: true, target_amp: 0.3 })
|
|
59
|
+
|
|
60
|
+
# Calling the group to play it
|
|
61
|
+
call myGroup
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
// Aggregator test: include tests from tests/rust/* as distinct modules.
|
|
2
|
+
// This keeps sources organized while providing a single integration test
|
|
3
|
+
// file that Cargo will compile and run.
|
|
4
|
+
|
|
5
|
+
mod rust_tests {
|
|
6
|
+
pub mod cli_help {
|
|
7
|
+
include!("rust/cli_help.rs");
|
|
8
|
+
}
|
|
9
|
+
pub mod cli_version {
|
|
10
|
+
include!("rust/cli_version.rs");
|
|
11
|
+
}
|
|
12
|
+
pub mod cli_template_list {
|
|
13
|
+
include!("rust/cli_template_list.rs");
|
|
14
|
+
}
|
|
15
|
+
pub mod cli_check_build {
|
|
16
|
+
include!("rust/cli_check_build.rs");
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
// Re-export tests so the test runner finds them at crate root.
|
|
21
|
+
pub use rust_tests::*;
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
use assert_cmd::Command;
|
|
2
|
+
use tempfile::tempdir;
|
|
3
|
+
|
|
4
|
+
#[test]
|
|
5
|
+
fn cli_check_and_build_no_crash() {
|
|
6
|
+
// create a temp output dir and run check/build against examples entry
|
|
7
|
+
let out = tempdir().expect("create tempdir");
|
|
8
|
+
let out_path = out.path().to_str().unwrap();
|
|
9
|
+
|
|
10
|
+
// Run `devalang check --entry examples --output <out> --debug`
|
|
11
|
+
let mut cmd = Command::cargo_bin("devalang").expect("binary not found");
|
|
12
|
+
cmd.args(["check", "--entry", "examples", "--output", out_path, "--debug"]);
|
|
13
|
+
let assert = cmd.assert();
|
|
14
|
+
// check may return success or non-zero depending on environment, ensure it doesn't panic
|
|
15
|
+
assert.get_output();
|
|
16
|
+
|
|
17
|
+
// Run build command similarly; we only assert it runs (not necessarily success on all envs)
|
|
18
|
+
let mut cmd2 = Command::cargo_bin("devalang").expect("binary not found");
|
|
19
|
+
cmd2.args(["build", "--entry", "examples", "--output", out_path, "--debug"]);
|
|
20
|
+
cmd2.assert().get_output();
|
|
21
|
+
}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
use assert_cmd::Command;
|
|
2
|
+
use predicates::prelude::*;
|
|
3
|
+
|
|
4
|
+
#[test]
|
|
5
|
+
fn cli_prints_help() {
|
|
6
|
+
// Build the binary in debug mode and run it with --help
|
|
7
|
+
let mut cmd = Command::cargo_bin("devalang").expect("binary not found");
|
|
8
|
+
cmd.arg("--help");
|
|
9
|
+
cmd.assert()
|
|
10
|
+
.success()
|
|
11
|
+
.stdout(predicate::str::contains("devalang"));
|
|
12
|
+
}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
use assert_cmd::Command;
|
|
2
|
+
|
|
3
|
+
#[test]
|
|
4
|
+
fn cli_template_list_runs() {
|
|
5
|
+
let mut cmd = Command::cargo_bin("devalang").expect("binary not found");
|
|
6
|
+
cmd.args(["template", "list"]);
|
|
7
|
+
let assert = cmd.assert();
|
|
8
|
+
// ensure it runs without panic; output may vary
|
|
9
|
+
assert.get_output();
|
|
10
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
use assert_cmd::Command;
|
|
2
|
+
use predicates::prelude::*;
|
|
3
|
+
|
|
4
|
+
#[test]
|
|
5
|
+
fn cli_prints_version() {
|
|
6
|
+
let mut cmd = Command::cargo_bin("devalang").expect("binary not found");
|
|
7
|
+
cmd.arg("--version");
|
|
8
|
+
cmd.assert()
|
|
9
|
+
.success()
|
|
10
|
+
.stdout(predicate::str::contains("devalang"));
|
|
11
|
+
}
|
|
@@ -0,0 +1,136 @@
|
|
|
1
|
+
import { expect } from "chai";
|
|
2
|
+
import * as fs from "fs";
|
|
3
|
+
import * as os from "os";
|
|
4
|
+
import * as path from "path";
|
|
5
|
+
|
|
6
|
+
// CommonJS bundle import (TypeScript-friendly)
|
|
7
|
+
const core: any = require("../../out-tsc");
|
|
8
|
+
|
|
9
|
+
describe("devalang typescript index", () => {
|
|
10
|
+
let origCwd: string;
|
|
11
|
+
let tmpDir: string | null = null;
|
|
12
|
+
|
|
13
|
+
before(function () {
|
|
14
|
+
// Prepare a temporary project root so wasm code that looks up project files
|
|
15
|
+
// and `.devalang` config can succeed during E2E runs.
|
|
16
|
+
origCwd = process.cwd();
|
|
17
|
+
tmpDir = fs.mkdtempSync(path.join(os.tmpdir(), "devalang-e2e-"));
|
|
18
|
+
// Create minimal .devalang config file so get_project_root finds it
|
|
19
|
+
fs.writeFileSync(
|
|
20
|
+
path.join(tmpDir, ".devalang"),
|
|
21
|
+
'name = "devalang-test"\n'
|
|
22
|
+
);
|
|
23
|
+
// Ensure .deva exists (optional banks/plugins)
|
|
24
|
+
fs.mkdirSync(path.join(tmpDir, ".deva"), { recursive: true });
|
|
25
|
+
process.chdir(tmpDir);
|
|
26
|
+
});
|
|
27
|
+
|
|
28
|
+
after(function () {
|
|
29
|
+
// restore cwd and attempt cleanup
|
|
30
|
+
if (tmpDir) {
|
|
31
|
+
try {
|
|
32
|
+
process.chdir(origCwd);
|
|
33
|
+
// best-effort cleanup
|
|
34
|
+
fs.rmSync(tmpDir, { recursive: true, force: true });
|
|
35
|
+
} catch (e) {
|
|
36
|
+
// ignore
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
it("should expose render_audio and return a Float32Array when wasm is available", function () {
|
|
41
|
+
// Prefer wasm binding if available
|
|
42
|
+
const wasmFn = core.pkg && core.pkg.render_audio;
|
|
43
|
+
|
|
44
|
+
const sampleProgram = `# simple program for tests
|
|
45
|
+
bpm 120
|
|
46
|
+
|
|
47
|
+
group myLead:
|
|
48
|
+
let mySynth = synth sine
|
|
49
|
+
mySynth -> note(C4, { duration: 400 })
|
|
50
|
+
mySynth -> note(G4, { duration: 400 })
|
|
51
|
+
call myLead
|
|
52
|
+
`;
|
|
53
|
+
|
|
54
|
+
if (typeof wasmFn === "function") {
|
|
55
|
+
try {
|
|
56
|
+
// Get parsed AST first for debugging
|
|
57
|
+
if (core.pkg && typeof core.pkg.parse === "function") {
|
|
58
|
+
try {
|
|
59
|
+
const parsed = core.pkg.parse("playground.deva", sampleProgram);
|
|
60
|
+
// parsed may be an object { ok, ast, errors } or a string
|
|
61
|
+
const astStr =
|
|
62
|
+
parsed && parsed.ast
|
|
63
|
+
? parsed.ast
|
|
64
|
+
: typeof parsed === "string"
|
|
65
|
+
? parsed
|
|
66
|
+
: JSON.stringify(parsed);
|
|
67
|
+
expect(astStr).to.match(/note|synth/i);
|
|
68
|
+
} catch (e) {
|
|
69
|
+
console.warn(
|
|
70
|
+
"[TEST DEBUG] parse() failed or returned non-standard result:",
|
|
71
|
+
e
|
|
72
|
+
);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
const result = wasmFn(sampleProgram);
|
|
77
|
+
expect(result.constructor && result.constructor.name).to.match(
|
|
78
|
+
/Float32Array|Float64Array|ArrayBuffer|Uint8Array/
|
|
79
|
+
);
|
|
80
|
+
} catch (err: any) {
|
|
81
|
+
// If the wasm panicked because the audio buffer is empty, skip the test in this env.
|
|
82
|
+
const msg =
|
|
83
|
+
typeof err === "string"
|
|
84
|
+
? err
|
|
85
|
+
: err && err.message
|
|
86
|
+
? err.message
|
|
87
|
+
: String(err);
|
|
88
|
+
if (msg.includes("Audio buffer is empty")) {
|
|
89
|
+
// skip this test on environments where generating audio isn't supported
|
|
90
|
+
// eslint-disable-next-line no-invalid-this
|
|
91
|
+
this.skip();
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
throw err;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
99
|
+
|
|
100
|
+
// If wasm binding isn't available, ensure a runtime placeholder exists but don't call it
|
|
101
|
+
const runtimeFn = core.render_audio;
|
|
102
|
+
expect(runtimeFn).to.be.a("function");
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
it("should find statements for entry module", function () {
|
|
106
|
+
const parseFn = core.pkg && core.pkg.parse;
|
|
107
|
+
if (typeof parseFn !== "function") {
|
|
108
|
+
// nothing to test if parse binding missing
|
|
109
|
+
// eslint-disable-next-line no-invalid-this
|
|
110
|
+
this.skip();
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
try {
|
|
114
|
+
const res = parseFn("playground.deva", "bpm 120");
|
|
115
|
+
expect(res).to.be.an("object");
|
|
116
|
+
// prefer explicit shape when available
|
|
117
|
+
if (Object.prototype.hasOwnProperty.call(res, "ok")) {
|
|
118
|
+
expect(res.ok).to.equal(true);
|
|
119
|
+
} else {
|
|
120
|
+
expect(res).to.have.property("ast");
|
|
121
|
+
}
|
|
122
|
+
} catch (err: any) {
|
|
123
|
+
const msg = err && err.message ? err.message : String(err);
|
|
124
|
+
if (
|
|
125
|
+
msg.includes("Module loading error") ||
|
|
126
|
+
msg.includes("Project root not found")
|
|
127
|
+
) {
|
|
128
|
+
// environment not prepared for module loading, skip
|
|
129
|
+
// eslint-disable-next-line no-invalid-this
|
|
130
|
+
this.skip();
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
throw err;
|
|
134
|
+
}
|
|
135
|
+
});
|
|
136
|
+
});
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { expect } from "chai";
|
|
2
|
+
|
|
3
|
+
const core: any = require("../../out-tsc");
|
|
4
|
+
|
|
5
|
+
describe("playhead callback", () => {
|
|
6
|
+
it("should call registered callback during run", () => {
|
|
7
|
+
const register = core.pkg && core.pkg.register_playhead_callback;
|
|
8
|
+
const unregister = core.pkg && core.pkg.unregister_playhead_callback;
|
|
9
|
+
if (typeof register !== "function") {
|
|
10
|
+
// skip when wasm not available
|
|
11
|
+
return;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
let called = false;
|
|
15
|
+
const cb = (ev: any) => {
|
|
16
|
+
called = true;
|
|
17
|
+
expect(ev).to.have.property("time");
|
|
18
|
+
expect(ev).to.have.property("line");
|
|
19
|
+
expect(ev).to.have.property("column");
|
|
20
|
+
};
|
|
21
|
+
|
|
22
|
+
register(cb);
|
|
23
|
+
|
|
24
|
+
// run a simple program to trigger events
|
|
25
|
+
const program = `bpm 120\n\n.myTrigger 1/4\n`;
|
|
26
|
+
// render_audio will schedule triggers and should invoke the callback during execution
|
|
27
|
+
const render = core.pkg && core.pkg.render_audio;
|
|
28
|
+
if (typeof render === "function") {
|
|
29
|
+
render(program);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
unregister && unregister();
|
|
33
|
+
|
|
34
|
+
expect(called).to.be.true;
|
|
35
|
+
});
|
|
36
|
+
});
|