@devaloop/devalang 0.0.1-beta.1 → 0.0.1-beta.3
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/.devalang +9 -10
- package/Cargo.toml +84 -80
- package/README.md +10 -7
- package/docs/CHANGELOG.md +83 -0
- package/docs/ROADMAP.md +6 -2
- package/docs/TODO.md +3 -14
- package/examples/bus.deva +10 -0
- package/examples/chain.deva +19 -0
- package/examples/effect.deva +2 -0
- package/examples/filter.deva +11 -0
- package/examples/lfo.deva +9 -0
- package/examples/plugin.deva +10 -10
- package/examples/routing.deva +23 -0
- package/examples/synth.deva +11 -1
- package/examples/synth_types.deva +17 -0
- package/out-tsc/bin/project-version.json +6 -0
- package/out-tsc/core/functions/index.d.ts +5 -0
- package/out-tsc/core/functions/index.js +11 -0
- package/out-tsc/pkg/devalang_core.d.ts +2 -0
- package/out-tsc/pkg/devalang_core.js +17 -2
- package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +1 -0
- package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
- package/out-tsc/scripts/version/copy-to-binary.js +79 -0
- package/package.json +23 -10
- package/project-version.json +3 -3
- package/rust/bindings/Cargo.toml +9 -0
- package/rust/bindings/src/lib.rs +86 -0
- package/rust/cli/addon/commands.rs +35 -0
- package/rust/cli/addon/download.rs +234 -0
- package/rust/cli/addon/install.rs +33 -0
- package/rust/cli/addon/list.rs +224 -0
- package/rust/cli/addon/metadata.rs +124 -0
- package/rust/cli/addon/mod.rs +8 -0
- package/rust/cli/addon/remove.rs +271 -0
- package/rust/cli/addon/update.rs +305 -0
- package/rust/cli/{install/addon.rs → addon/utils.rs} +34 -43
- package/rust/cli/build/commands.rs +153 -103
- package/rust/cli/build/mod.rs +2 -2
- package/rust/cli/build/process.rs +165 -146
- package/rust/cli/check/mod.rs +208 -208
- package/rust/cli/discover/commands.rs +53 -31
- package/rust/cli/discover/config.rs +2 -4
- package/rust/cli/discover/install.rs +139 -28
- package/rust/cli/discover/metadata.rs +3 -3
- package/rust/cli/login/commands.rs +124 -124
- package/rust/cli/me/commands.rs +52 -0
- package/rust/cli/me/mod.rs +1 -0
- package/rust/cli/mod.rs +2 -2
- package/rust/cli/parser.rs +76 -70
- package/rust/cli/play/commands.rs +375 -324
- package/rust/cli/play/mod.rs +5 -5
- package/rust/cli/play/process.rs +159 -150
- package/rust/cli/play/realtime.rs +91 -91
- package/rust/cli/telemetry/commands.rs +22 -22
- package/rust/cli/telemetry/event_creator.rs +80 -80
- package/rust/cli/telemetry/mod.rs +3 -3
- package/rust/cli/telemetry/send.rs +51 -51
- package/rust/cli/template/commands.rs +69 -69
- package/rust/config/driver.rs +112 -103
- package/rust/config/mod.rs +3 -3
- package/rust/config/ops.rs +26 -26
- package/rust/config/settings.rs +101 -101
- package/rust/core/audio/engine/driver.rs +237 -0
- package/rust/core/audio/engine/export.rs +169 -0
- package/rust/core/audio/engine/helpers.rs +178 -170
- package/rust/core/audio/engine/mod.rs +56 -7
- package/rust/core/audio/engine/notes/dsp.rs +88 -0
- package/rust/core/audio/engine/notes/mod.rs +53 -0
- package/rust/core/audio/engine/notes/params.rs +294 -0
- package/rust/core/audio/engine/sample/insert.rs +300 -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.rs → evaluator/numeric.rs} +152 -310
- 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 -542
- package/rust/core/audio/interpreter/mod.rs +2 -14
- package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +179 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +3 -0
- package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +371 -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/{automate.rs → statements/automate.rs} +2 -4
- package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +36 -5
- package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -71
- package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +24 -26
- package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +36 -38
- package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +17 -19
- package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +115 -114
- package/rust/core/audio/interpreter/statements/mod.rs +12 -0
- package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
- package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +54 -4
- package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
- package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +242 -239
- package/rust/core/audio/loader/trigger.rs +98 -97
- package/rust/core/audio/mod.rs +6 -7
- package/rust/core/audio/special/easing.rs +189 -189
- package/rust/core/audio/special/env.rs +45 -45
- package/rust/core/audio/special/math.rs +134 -134
- package/rust/core/audio/special/modulator.rs +143 -143
- package/rust/core/builder/mod.rs +129 -86
- package/rust/core/debugger/{module.rs → logs.rs} +52 -55
- package/rust/core/debugger/mod.rs +30 -30
- package/rust/core/debugger/store.rs +38 -40
- package/rust/core/error/mod.rs +269 -269
- package/rust/core/lexer/driver.rs +2 -4
- 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 +159 -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 +247 -0
- package/rust/core/parser/driver/parser.rs +160 -0
- package/rust/core/parser/handler/arrow_call.rs +90 -15
- package/rust/core/parser/handler/at.rs +279 -279
- package/rust/core/parser/handler/bank.rs +104 -104
- package/rust/core/parser/handler/condition.rs +83 -83
- package/rust/core/parser/handler/dot.rs +148 -148
- package/rust/core/parser/handler/identifier/automate.rs +254 -254
- package/rust/core/parser/handler/identifier/call.rs +91 -91
- package/rust/core/parser/handler/identifier/emit.rs +70 -70
- package/rust/core/parser/handler/identifier/function.rs +113 -113
- package/rust/core/parser/handler/identifier/group.rs +89 -89
- package/rust/core/parser/handler/identifier/let_.rs +173 -173
- package/rust/core/parser/handler/identifier/mod.rs +55 -55
- package/rust/core/parser/handler/identifier/on.rs +107 -107
- package/rust/core/parser/handler/identifier/print.rs +49 -49
- package/rust/core/parser/handler/identifier/sleep.rs +96 -43
- package/rust/core/parser/handler/identifier/spawn.rs +91 -91
- package/rust/core/parser/handler/identifier/synth.rs +39 -3
- package/rust/core/parser/handler/loop_.rs +194 -194
- package/rust/core/parser/handler/pattern.rs +25 -2
- package/rust/core/parser/handler/tempo.rs +105 -57
- package/rust/core/parser/statement.rs +10 -11
- package/rust/core/plugin/loader.rs +137 -137
- package/rust/core/plugin/runner/mod.rs +11 -0
- package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +206 -72
- package/rust/core/plugin/runner/wasm32.rs +44 -0
- package/rust/core/preprocessor/loader/inject.rs +313 -0
- package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
- package/rust/core/preprocessor/loader/mod.rs +235 -0
- package/rust/core/preprocessor/module.rs +55 -60
- package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +107 -114
- package/rust/core/preprocessor/processor/mod.rs +1 -0
- package/rust/core/preprocessor/resolver/function.rs +69 -69
- package/rust/core/preprocessor/resolver/group.rs +122 -94
- package/rust/core/preprocessor/resolver/pattern.rs +14 -2
- package/rust/core/store/global.rs +57 -61
- package/rust/core/store/mod.rs +1 -5
- package/rust/lib.rs +323 -308
- package/rust/macros/Cargo.toml +14 -0
- package/rust/macros/src/lib.rs +52 -0
- package/rust/main.rs +336 -143
- package/rust/types/Cargo.toml +1 -1
- package/rust/types/src/addons.rs +57 -55
- package/rust/types/src/config.rs +82 -74
- package/rust/types/src/lib.rs +15 -12
- package/rust/types/src/plugin.rs +20 -0
- package/rust/types/src/store.rs +139 -0
- package/rust/types/src/telemetry.rs +85 -85
- package/rust/utils/Cargo.toml +5 -2
- package/rust/utils/src/file.rs +477 -94
- package/rust/utils/src/first_usage.rs +97 -97
- package/rust/utils/src/lib.rs +9 -9
- package/rust/utils/src/logger.rs +200 -200
- package/rust/utils/src/path.rs +158 -88
- package/rust/utils/src/signature.rs +41 -41
- package/rust/utils/src/spinner.rs +20 -20
- package/rust/utils/src/version.rs +58 -27
- package/rust/utils/src/watcher.rs +46 -46
- package/rust/web/api.rs +5 -5
- package/rust/web/auth.rs +5 -0
- package/rust/web/cdn.rs +34 -34
- package/rust/web/forge.rs +5 -0
- package/rust/web/mod.rs +2 -0
- package/tests/integration.rs +21 -21
- package/typescript/core/functions/index.ts +11 -0
- package/typescript/pkg/devalang_core.ts +20 -4
- package/typescript/scripts/version/copy-to-binary.ts +82 -0
- package/rust/cli/bank/api.rs +0 -122
- package/rust/cli/bank/commands.rs +0 -275
- package/rust/cli/bank/mod.rs +0 -29
- package/rust/cli/install/bank.rs +0 -53
- package/rust/cli/install/commands.rs +0 -35
- package/rust/cli/install/mod.rs +0 -4
- package/rust/cli/install/plugin.rs +0 -61
- package/rust/core/audio/engine/sample.rs +0 -366
- package/rust/core/audio/engine/synth.rs +0 -325
- package/rust/core/audio/interpreter/arrow_call.rs +0 -311
- package/rust/core/audio/renderer.rs +0 -54
- package/rust/core/parser/driver.rs +0 -584
- package/rust/core/preprocessor/loader.rs +0 -637
- package/rust/core/store/export.rs +0 -28
- package/rust/core/store/function.rs +0 -40
- package/rust/core/store/import.rs +0 -28
- package/rust/core/store/variable.rs +0 -51
- package/rust/core/utils/mod.rs +0 -1
- package/rust/core/utils/path.rs +0 -37
package/rust/config/ops.rs
CHANGED
|
@@ -1,26 +1,26 @@
|
|
|
1
|
-
use crate::config::driver::ProjectConfig;
|
|
2
|
-
use std::fs;
|
|
3
|
-
use std::path::Path;
|
|
4
|
-
|
|
5
|
-
pub fn load_config(path: Option<&Path>) -> Option<ProjectConfig> {
|
|
6
|
-
let config_path_buf;
|
|
7
|
-
let config_path = match path {
|
|
8
|
-
Some(p) => p,
|
|
9
|
-
None => {
|
|
10
|
-
config_path_buf = match devalang_utils::path::get_devalang_config_path() {
|
|
11
|
-
Ok(p) => p,
|
|
12
|
-
Err(_) => {
|
|
13
|
-
return None;
|
|
14
|
-
}
|
|
15
|
-
};
|
|
16
|
-
&config_path_buf
|
|
17
|
-
}
|
|
18
|
-
};
|
|
19
|
-
|
|
20
|
-
if config_path.exists() {
|
|
21
|
-
let content = fs::read_to_string(config_path).ok()?;
|
|
22
|
-
toml::from_str(&content).ok()
|
|
23
|
-
} else {
|
|
24
|
-
None
|
|
25
|
-
}
|
|
26
|
-
}
|
|
1
|
+
use crate::config::driver::ProjectConfig;
|
|
2
|
+
use std::fs;
|
|
3
|
+
use std::path::Path;
|
|
4
|
+
|
|
5
|
+
pub fn load_config(path: Option<&Path>) -> Option<ProjectConfig> {
|
|
6
|
+
let config_path_buf;
|
|
7
|
+
let config_path = match path {
|
|
8
|
+
Some(p) => p,
|
|
9
|
+
None => {
|
|
10
|
+
config_path_buf = match devalang_utils::path::get_devalang_config_path() {
|
|
11
|
+
Ok(p) => p,
|
|
12
|
+
Err(_) => {
|
|
13
|
+
return None;
|
|
14
|
+
}
|
|
15
|
+
};
|
|
16
|
+
&config_path_buf
|
|
17
|
+
}
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
if config_path.exists() {
|
|
21
|
+
let content = fs::read_to_string(config_path).ok()?;
|
|
22
|
+
toml::from_str(&content).ok()
|
|
23
|
+
} else {
|
|
24
|
+
None
|
|
25
|
+
}
|
|
26
|
+
}
|
package/rust/config/settings.rs
CHANGED
|
@@ -1,101 +1,101 @@
|
|
|
1
|
-
use devalang_types::{TelemetrySettings, UserSettings};
|
|
2
|
-
use serde_json::Value as JsonValue;
|
|
3
|
-
use std::io::Write;
|
|
4
|
-
|
|
5
|
-
pub fn get_home_dir() -> Option<std::path::PathBuf> {
|
|
6
|
-
dirs::home_dir()
|
|
7
|
-
}
|
|
8
|
-
|
|
9
|
-
pub fn get_devalang_homedir() -> std::path::PathBuf {
|
|
10
|
-
if let Some(home_dir) = get_home_dir() {
|
|
11
|
-
home_dir.join(".devalang")
|
|
12
|
-
} else {
|
|
13
|
-
std::path::PathBuf::from("~/.devalang")
|
|
14
|
-
}
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
pub fn get_default_user_config() -> UserSettings {
|
|
18
|
-
UserSettings {
|
|
19
|
-
session: "".into(),
|
|
20
|
-
telemetry: TelemetrySettings {
|
|
21
|
-
uuid: uuid::Uuid::new_v4().to_string(),
|
|
22
|
-
enabled: false,
|
|
23
|
-
level: "basic".into(),
|
|
24
|
-
stats: false,
|
|
25
|
-
},
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
pub fn get_user_config() -> Option<UserSettings> {
|
|
30
|
-
if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
|
|
31
|
-
let file = std::fs::File::open(config_path).ok()?;
|
|
32
|
-
let settings = serde_json::from_reader(file).ok()?;
|
|
33
|
-
Some(settings)
|
|
34
|
-
} else {
|
|
35
|
-
None
|
|
36
|
-
}
|
|
37
|
-
}
|
|
38
|
-
|
|
39
|
-
pub fn write_user_config_file() {
|
|
40
|
-
if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
|
|
41
|
-
let settings = get_user_config().unwrap_or_else(get_default_user_config);
|
|
42
|
-
|
|
43
|
-
let config_json = serde_json::to_string(&settings).unwrap();
|
|
44
|
-
|
|
45
|
-
if let Err(e) = write_config_atomic(&config_path, &config_json) {
|
|
46
|
-
println!("Could not write config file: {}", e);
|
|
47
|
-
}
|
|
48
|
-
} else {
|
|
49
|
-
println!("Could not create config file");
|
|
50
|
-
}
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
pub fn ensure_user_config_file_exists() {
|
|
54
|
-
if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
|
|
55
|
-
if !config_path.exists() {
|
|
56
|
-
write_user_config_file();
|
|
57
|
-
}
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
pub fn set_user_config_value(key: &str, value: JsonValue) {
|
|
62
|
-
let mut settings = get_user_config().unwrap_or_default();
|
|
63
|
-
|
|
64
|
-
match (key, &value) {
|
|
65
|
-
("telemetry", JsonValue::Bool(b)) => {
|
|
66
|
-
settings.telemetry.enabled = *b;
|
|
67
|
-
}
|
|
68
|
-
("session", JsonValue::String(s)) => {
|
|
69
|
-
settings.session = s.clone();
|
|
70
|
-
}
|
|
71
|
-
_ => {
|
|
72
|
-
println!("Unsupported key or value type for '{}': {:?}", key, value);
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
|
|
77
|
-
let config_json = serde_json::to_string(&settings).unwrap();
|
|
78
|
-
if let Err(e) = write_config_atomic(&config_path, &config_json) {
|
|
79
|
-
println!("Could not write config file: {}", e);
|
|
80
|
-
}
|
|
81
|
-
} else {
|
|
82
|
-
println!("Could not create config file");
|
|
83
|
-
}
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
pub fn write_config_atomic(
|
|
87
|
-
config_path: &std::path::PathBuf,
|
|
88
|
-
contents: &str,
|
|
89
|
-
) -> std::io::Result<()> {
|
|
90
|
-
if let Some(parent) = config_path.parent() {
|
|
91
|
-
std::fs::create_dir_all(parent)?;
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
let tmp_path = config_path.with_extension("json.tmp");
|
|
95
|
-
let mut tmp_file = std::fs::File::create(&tmp_path)?;
|
|
96
|
-
tmp_file.write_all(contents.as_bytes())?;
|
|
97
|
-
tmp_file.sync_all()?;
|
|
98
|
-
std::fs::rename(&tmp_path, config_path)?;
|
|
99
|
-
|
|
100
|
-
Ok(())
|
|
101
|
-
}
|
|
1
|
+
use devalang_types::{TelemetrySettings, UserSettings};
|
|
2
|
+
use serde_json::Value as JsonValue;
|
|
3
|
+
use std::io::Write;
|
|
4
|
+
|
|
5
|
+
pub fn get_home_dir() -> Option<std::path::PathBuf> {
|
|
6
|
+
dirs::home_dir()
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
pub fn get_devalang_homedir() -> std::path::PathBuf {
|
|
10
|
+
if let Some(home_dir) = get_home_dir() {
|
|
11
|
+
home_dir.join(".devalang")
|
|
12
|
+
} else {
|
|
13
|
+
std::path::PathBuf::from("~/.devalang")
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
pub fn get_default_user_config() -> UserSettings {
|
|
18
|
+
UserSettings {
|
|
19
|
+
session: "".into(),
|
|
20
|
+
telemetry: TelemetrySettings {
|
|
21
|
+
uuid: uuid::Uuid::new_v4().to_string(),
|
|
22
|
+
enabled: false,
|
|
23
|
+
level: "basic".into(),
|
|
24
|
+
stats: false,
|
|
25
|
+
},
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
pub fn get_user_config() -> Option<UserSettings> {
|
|
30
|
+
if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
|
|
31
|
+
let file = std::fs::File::open(config_path).ok()?;
|
|
32
|
+
let settings = serde_json::from_reader(file).ok()?;
|
|
33
|
+
Some(settings)
|
|
34
|
+
} else {
|
|
35
|
+
None
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
pub fn write_user_config_file() {
|
|
40
|
+
if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
|
|
41
|
+
let settings = get_user_config().unwrap_or_else(get_default_user_config);
|
|
42
|
+
|
|
43
|
+
let config_json = serde_json::to_string(&settings).unwrap();
|
|
44
|
+
|
|
45
|
+
if let Err(e) = write_config_atomic(&config_path, &config_json) {
|
|
46
|
+
println!("Could not write config file: {}", e);
|
|
47
|
+
}
|
|
48
|
+
} else {
|
|
49
|
+
println!("Could not create config file");
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
pub fn ensure_user_config_file_exists() {
|
|
54
|
+
if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
|
|
55
|
+
if !config_path.exists() {
|
|
56
|
+
write_user_config_file();
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
pub fn set_user_config_value(key: &str, value: JsonValue) {
|
|
62
|
+
let mut settings = get_user_config().unwrap_or_default();
|
|
63
|
+
|
|
64
|
+
match (key, &value) {
|
|
65
|
+
("telemetry", JsonValue::Bool(b)) => {
|
|
66
|
+
settings.telemetry.enabled = *b;
|
|
67
|
+
}
|
|
68
|
+
("session", JsonValue::String(s)) => {
|
|
69
|
+
settings.session = s.clone();
|
|
70
|
+
}
|
|
71
|
+
_ => {
|
|
72
|
+
println!("Unsupported key or value type for '{}': {:?}", key, value);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
if let Some(config_path) = get_devalang_homedir().join("config.json").into() {
|
|
77
|
+
let config_json = serde_json::to_string(&settings).unwrap();
|
|
78
|
+
if let Err(e) = write_config_atomic(&config_path, &config_json) {
|
|
79
|
+
println!("Could not write config file: {}", e);
|
|
80
|
+
}
|
|
81
|
+
} else {
|
|
82
|
+
println!("Could not create config file");
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
pub fn write_config_atomic(
|
|
87
|
+
config_path: &std::path::PathBuf,
|
|
88
|
+
contents: &str,
|
|
89
|
+
) -> std::io::Result<()> {
|
|
90
|
+
if let Some(parent) = config_path.parent() {
|
|
91
|
+
std::fs::create_dir_all(parent)?;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
let tmp_path = config_path.with_extension("json.tmp");
|
|
95
|
+
let mut tmp_file = std::fs::File::create(&tmp_path)?;
|
|
96
|
+
tmp_file.write_all(contents.as_bytes())?;
|
|
97
|
+
tmp_file.sync_all()?;
|
|
98
|
+
std::fs::rename(&tmp_path, config_path)?;
|
|
99
|
+
|
|
100
|
+
Ok(())
|
|
101
|
+
}
|
|
@@ -0,0 +1,237 @@
|
|
|
1
|
+
use devalang_types::Value;
|
|
2
|
+
use std::collections::HashMap;
|
|
3
|
+
|
|
4
|
+
// Minimal representation of a MIDI note event for export purposes.
|
|
5
|
+
#[derive(Debug, Clone, PartialEq)]
|
|
6
|
+
pub struct MidiNoteEvent {
|
|
7
|
+
/// MIDI key number 0-127
|
|
8
|
+
pub key: u8,
|
|
9
|
+
/// velocity 0-127
|
|
10
|
+
pub vel: u8,
|
|
11
|
+
/// start time in milliseconds (absolute)
|
|
12
|
+
pub start_ms: u32,
|
|
13
|
+
/// duration in milliseconds
|
|
14
|
+
pub duration_ms: u32,
|
|
15
|
+
/// MIDI channel (0-15)
|
|
16
|
+
pub channel: u8,
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
// Sample rate and channel constants used throughout the engine.
|
|
20
|
+
pub const SAMPLE_RATE: u32 = 44100;
|
|
21
|
+
pub const CHANNELS: u16 = 2;
|
|
22
|
+
|
|
23
|
+
/// AudioEngine holds the generated interleaved stereo buffer and
|
|
24
|
+
/// provides simple utilities to mix/merge buffers and export WAV files.
|
|
25
|
+
///
|
|
26
|
+
/// Notes:
|
|
27
|
+
/// - Buffer is interleaved stereo (L,R,L,R...).
|
|
28
|
+
/// - Methods are synchronous and operate on in-memory buffers.
|
|
29
|
+
///
|
|
30
|
+
#[derive(Debug, Clone, PartialEq)]
|
|
31
|
+
pub struct AudioEngine {
|
|
32
|
+
/// Master volume multiplier (not automatically applied by helpers).
|
|
33
|
+
pub volume: f32,
|
|
34
|
+
/// Interleaved i16 PCM buffer.
|
|
35
|
+
pub buffer: Vec<i16>,
|
|
36
|
+
/// Collected MIDI note events for export (non-audio representation).
|
|
37
|
+
pub midi_events: Vec<MidiNoteEvent>,
|
|
38
|
+
/// Map target synth -> last inserted note sample ranges (start_sample, total_samples)
|
|
39
|
+
pub last_notes: std::collections::HashMap<String, Vec<(usize, usize)>>,
|
|
40
|
+
/// Logical module name used for error traces/diagnostics.
|
|
41
|
+
pub module_name: String,
|
|
42
|
+
/// Simple diagnostic counter for inserted notes.
|
|
43
|
+
pub note_count: usize,
|
|
44
|
+
/// Sample rate (can be overridden per-engine)
|
|
45
|
+
pub sample_rate: u32,
|
|
46
|
+
/// Number of channels (interleaved). Defaults to 2.
|
|
47
|
+
pub channels: u16,
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
impl AudioEngine {
|
|
51
|
+
pub fn new(module_name: String) -> Self {
|
|
52
|
+
AudioEngine {
|
|
53
|
+
volume: 1.0,
|
|
54
|
+
buffer: vec![],
|
|
55
|
+
midi_events: Vec::new(),
|
|
56
|
+
module_name,
|
|
57
|
+
note_count: 0,
|
|
58
|
+
sample_rate: SAMPLE_RATE,
|
|
59
|
+
channels: CHANNELS,
|
|
60
|
+
last_notes: std::collections::HashMap::new(),
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
pub fn get_buffer(&self) -> &[i16] {
|
|
65
|
+
&self.buffer
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
pub fn get_normalized_buffer(&self) -> Vec<f32> {
|
|
69
|
+
self.buffer.iter().map(|&s| (s as f32) / 32768.0).collect()
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
pub fn mix(&mut self, other: &AudioEngine) {
|
|
73
|
+
let max_len = self.buffer.len().max(other.buffer.len());
|
|
74
|
+
self.buffer.resize(max_len, 0);
|
|
75
|
+
|
|
76
|
+
for (i, &sample) in other.buffer.iter().enumerate() {
|
|
77
|
+
self.buffer[i] = self.buffer[i].saturating_add(sample);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
pub fn merge_with(&mut self, other: AudioEngine) {
|
|
82
|
+
// If the other buffer is empty, simply return without warning (common for spawns that produced nothing)
|
|
83
|
+
if other.buffer.is_empty() {
|
|
84
|
+
return;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
// If the other buffer is present but contains only zeros, warn and skip merge
|
|
88
|
+
if other.buffer.iter().all(|&s| s == 0) {
|
|
89
|
+
eprintln!("⚠️ Skipping merge: other buffer is silent");
|
|
90
|
+
return;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
if self.buffer.iter().all(|&s| s == 0) {
|
|
94
|
+
self.buffer = other.buffer;
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
self.mix(&other);
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
pub fn set_duration(&mut self, duration_secs: f32) {
|
|
102
|
+
let total_samples =
|
|
103
|
+
(duration_secs * (self.sample_rate as f32) * (self.channels as f32)) as usize;
|
|
104
|
+
|
|
105
|
+
if self.buffer.len() < total_samples {
|
|
106
|
+
self.buffer.resize(total_samples, 0);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
pub fn generate_midi_file(
|
|
111
|
+
&mut self,
|
|
112
|
+
output_path: &String,
|
|
113
|
+
bpm: Option<f32>,
|
|
114
|
+
tpqn: Option<u16>,
|
|
115
|
+
) -> Result<(), String> {
|
|
116
|
+
crate::core::audio::engine::export::generate_midi_file_impl(
|
|
117
|
+
&self.midi_events,
|
|
118
|
+
output_path,
|
|
119
|
+
bpm,
|
|
120
|
+
tpqn,
|
|
121
|
+
)
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
pub fn generate_wav_file(
|
|
125
|
+
&mut self,
|
|
126
|
+
output_dir: &String,
|
|
127
|
+
audio_format: Option<String>,
|
|
128
|
+
sample_rate: Option<u32>,
|
|
129
|
+
) -> Result<(), String> {
|
|
130
|
+
crate::core::audio::engine::export::generate_wav_file_impl(
|
|
131
|
+
&mut self.buffer,
|
|
132
|
+
output_dir,
|
|
133
|
+
audio_format,
|
|
134
|
+
sample_rate,
|
|
135
|
+
)
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
pub fn insert_note(
|
|
139
|
+
&mut self,
|
|
140
|
+
owner: Option<String>,
|
|
141
|
+
waveform: String,
|
|
142
|
+
freq: f32,
|
|
143
|
+
amp: f32,
|
|
144
|
+
start_time_ms: f32,
|
|
145
|
+
duration_ms: f32,
|
|
146
|
+
synth_params: HashMap<String, Value>,
|
|
147
|
+
note_params: HashMap<String, Value>,
|
|
148
|
+
automation: Option<HashMap<String, Value>>,
|
|
149
|
+
) -> Vec<(usize, usize)> {
|
|
150
|
+
// Delegated implementation lives in notes.rs
|
|
151
|
+
crate::core::audio::engine::notes::insert_note_impl(
|
|
152
|
+
self,
|
|
153
|
+
owner,
|
|
154
|
+
waveform,
|
|
155
|
+
freq,
|
|
156
|
+
amp,
|
|
157
|
+
start_time_ms,
|
|
158
|
+
duration_ms,
|
|
159
|
+
synth_params,
|
|
160
|
+
note_params,
|
|
161
|
+
automation,
|
|
162
|
+
)
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
pub fn record_last_note_range(
|
|
166
|
+
&mut self,
|
|
167
|
+
owner: &str,
|
|
168
|
+
start_sample: usize,
|
|
169
|
+
total_samples: usize,
|
|
170
|
+
) {
|
|
171
|
+
self.last_notes
|
|
172
|
+
.entry(owner.to_string())
|
|
173
|
+
.or_default()
|
|
174
|
+
.push((start_sample, total_samples));
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
// helper extraction functions left in this struct for now
|
|
178
|
+
pub(crate) fn extract_f32(&self, map: &HashMap<String, Value>, key: &str) -> Option<f32> {
|
|
179
|
+
match map.get(key) {
|
|
180
|
+
Some(Value::Number(n)) => Some(*n),
|
|
181
|
+
Some(Value::String(s)) => s.parse::<f32>().ok(),
|
|
182
|
+
Some(Value::Boolean(b)) => Some(if *b { 1.0 } else { 0.0 }),
|
|
183
|
+
_ => None,
|
|
184
|
+
}
|
|
185
|
+
}
|
|
186
|
+
|
|
187
|
+
pub(crate) fn extract_boolean(&self, map: &HashMap<String, Value>, key: &str) -> Option<bool> {
|
|
188
|
+
match map.get(key) {
|
|
189
|
+
Some(Value::Boolean(b)) => Some(*b),
|
|
190
|
+
Some(Value::Number(n)) => Some(*n != 0.0),
|
|
191
|
+
Some(Value::Identifier(s)) => {
|
|
192
|
+
if s == "true" {
|
|
193
|
+
Some(true)
|
|
194
|
+
} else if s == "false" {
|
|
195
|
+
Some(false)
|
|
196
|
+
} else {
|
|
197
|
+
None
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
Some(Value::String(s)) => {
|
|
201
|
+
if s == "true" {
|
|
202
|
+
Some(true)
|
|
203
|
+
} else if s == "false" {
|
|
204
|
+
Some(false)
|
|
205
|
+
} else {
|
|
206
|
+
None
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
_ => None,
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
// Parse simple musical fraction strings like "1/16" into seconds using bpm
|
|
215
|
+
pub fn parse_fraction_to_seconds(s: &str, bpm: f32) -> Option<f32> {
|
|
216
|
+
let trimmed = s.trim();
|
|
217
|
+
if let Some((num, den)) = trimmed.split_once('/') {
|
|
218
|
+
if let (Ok(n), Ok(d)) = (num.parse::<f32>(), den.parse::<f32>()) {
|
|
219
|
+
if d != 0.0 {
|
|
220
|
+
let beats = n / d; // e.g. 1/16 -> 0.0625 beats
|
|
221
|
+
let secs_per_beat = 60.0 / bpm.max(1.0);
|
|
222
|
+
return Some(beats * secs_per_beat);
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
None
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
// Convert a devalang_types::Duration to seconds using bpm when relevant.
|
|
230
|
+
pub fn duration_to_seconds(d: &devalang_types::Duration, bpm: f32) -> Option<f32> {
|
|
231
|
+
use devalang_types::Duration as D;
|
|
232
|
+
match d {
|
|
233
|
+
D::Number(s) => Some(*s),
|
|
234
|
+
D::Beat(frac) | D::Identifier(frac) => parse_fraction_to_seconds(frac, bpm),
|
|
235
|
+
_ => None,
|
|
236
|
+
}
|
|
237
|
+
}
|
|
@@ -0,0 +1,169 @@
|
|
|
1
|
+
use crate::core::audio::engine::driver::MidiNoteEvent;
|
|
2
|
+
use std::fs::File;
|
|
3
|
+
|
|
4
|
+
pub fn generate_midi_file_impl(
|
|
5
|
+
midi_events: &Vec<MidiNoteEvent>,
|
|
6
|
+
output_path: &String,
|
|
7
|
+
bpm: Option<f32>,
|
|
8
|
+
tpqn: Option<u16>,
|
|
9
|
+
) -> Result<(), String> {
|
|
10
|
+
use midly::num::{u7, u15, u24};
|
|
11
|
+
use midly::{
|
|
12
|
+
Format, Header, MetaMessage, MidiMessage, Smf, Timing, TrackEvent, TrackEventKind,
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
if midi_events.is_empty() {
|
|
16
|
+
return Ok(());
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
let bpm = bpm.unwrap_or(120.0_f32);
|
|
20
|
+
let tpqn: u16 = tpqn.unwrap_or(480u16);
|
|
21
|
+
let header = Header {
|
|
22
|
+
format: Format::SingleTrack,
|
|
23
|
+
timing: Timing::Metrical(u15::from(tpqn)),
|
|
24
|
+
};
|
|
25
|
+
|
|
26
|
+
#[derive(Clone)]
|
|
27
|
+
struct AbsEvent {
|
|
28
|
+
tick: u64,
|
|
29
|
+
kind: TrackEventKind<'static>,
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
let mut abs_events: Vec<AbsEvent> = Vec::new();
|
|
33
|
+
let microsecs_per_quarter = (60_000_000.0 / bpm) as u32;
|
|
34
|
+
abs_events.push(AbsEvent {
|
|
35
|
+
tick: 0,
|
|
36
|
+
kind: TrackEventKind::Meta(MetaMessage::Tempo(u24::from(microsecs_per_quarter))),
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
for ev in midi_events {
|
|
40
|
+
let start_secs = (ev.start_ms as f32) / 1000.0;
|
|
41
|
+
let dur_secs = (ev.duration_ms as f32) / 1000.0;
|
|
42
|
+
let start_ticks_f = start_secs * (bpm / 60.0) * (tpqn as f32);
|
|
43
|
+
let dur_ticks_f = dur_secs * (bpm / 60.0) * (tpqn as f32);
|
|
44
|
+
let start_tick = start_ticks_f.max(0.0).round() as u64;
|
|
45
|
+
let off_tick = (start_ticks_f + dur_ticks_f).max(start_tick as f32).round() as u64;
|
|
46
|
+
|
|
47
|
+
let key = u7::from(ev.key.min(127));
|
|
48
|
+
let vel = u7::from(ev.vel.min(127));
|
|
49
|
+
|
|
50
|
+
abs_events.push(AbsEvent {
|
|
51
|
+
tick: start_tick,
|
|
52
|
+
kind: TrackEventKind::Midi {
|
|
53
|
+
channel: (ev.channel as u8).into(),
|
|
54
|
+
message: MidiMessage::NoteOn { key, vel },
|
|
55
|
+
},
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
abs_events.push(AbsEvent {
|
|
59
|
+
tick: off_tick,
|
|
60
|
+
kind: TrackEventKind::Midi {
|
|
61
|
+
channel: (ev.channel as u8).into(),
|
|
62
|
+
message: MidiMessage::NoteOff {
|
|
63
|
+
key,
|
|
64
|
+
vel: u7::from(0),
|
|
65
|
+
},
|
|
66
|
+
},
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
let max_tick = abs_events.iter().map(|e| e.tick).max().unwrap_or(0);
|
|
71
|
+
abs_events.push(AbsEvent {
|
|
72
|
+
tick: max_tick + 0,
|
|
73
|
+
kind: TrackEventKind::Meta(MetaMessage::EndOfTrack),
|
|
74
|
+
});
|
|
75
|
+
abs_events.sort_by_key(|e| e.tick);
|
|
76
|
+
|
|
77
|
+
let mut track: Vec<TrackEvent> = Vec::new();
|
|
78
|
+
let mut last_tick: u64 = 0;
|
|
79
|
+
for e in abs_events {
|
|
80
|
+
let delta = (e.tick - last_tick) as u32;
|
|
81
|
+
track.push(TrackEvent {
|
|
82
|
+
delta: delta.into(),
|
|
83
|
+
kind: e.kind,
|
|
84
|
+
});
|
|
85
|
+
last_tick = e.tick;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
let smf = Smf {
|
|
89
|
+
header,
|
|
90
|
+
tracks: vec![track],
|
|
91
|
+
};
|
|
92
|
+
|
|
93
|
+
if let Ok(mut file) = File::create(output_path) {
|
|
94
|
+
if let Err(e) = smf.write_std(&mut file) {
|
|
95
|
+
return Err(format!("Error writing MIDI file: {}", e));
|
|
96
|
+
}
|
|
97
|
+
} else {
|
|
98
|
+
return Err(format!("Cannot create MIDI file at {}", output_path));
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
Ok(())
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
pub fn generate_wav_file_impl(
|
|
105
|
+
buffer: &mut Vec<i16>,
|
|
106
|
+
output_dir: &String,
|
|
107
|
+
audio_format: Option<String>,
|
|
108
|
+
sample_rate: Option<u32>,
|
|
109
|
+
) -> Result<(), String> {
|
|
110
|
+
if buffer.len() % (crate::core::audio::engine::CHANNELS as usize) != 0 {
|
|
111
|
+
buffer.push(0);
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
let sr = sample_rate.unwrap_or(crate::core::audio::engine::SAMPLE_RATE);
|
|
115
|
+
let format_str = audio_format.unwrap_or_else(|| "Wav16".to_string());
|
|
116
|
+
let fmt_low = format_str.to_lowercase();
|
|
117
|
+
|
|
118
|
+
match fmt_low.as_str() {
|
|
119
|
+
"wav16" => {
|
|
120
|
+
let spec = hound::WavSpec {
|
|
121
|
+
channels: crate::core::audio::engine::CHANNELS,
|
|
122
|
+
sample_rate: sr,
|
|
123
|
+
bits_per_sample: 16,
|
|
124
|
+
sample_format: hound::SampleFormat::Int,
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
let mut writer = hound::WavWriter::create(output_dir, spec)
|
|
128
|
+
.map_err(|e| format!("Error creating WAV file: {}", e))?;
|
|
129
|
+
|
|
130
|
+
for sample in buffer.iter() {
|
|
131
|
+
writer
|
|
132
|
+
.write_sample(*sample)
|
|
133
|
+
.map_err(|e| format!("Error writing sample: {:?}", e))?;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
writer
|
|
137
|
+
.finalize()
|
|
138
|
+
.map_err(|e| format!("Error finalizing WAV: {:?}", e))?;
|
|
139
|
+
}
|
|
140
|
+
"wav24" | "wav32" => {
|
|
141
|
+
let bits = if fmt_low.contains("24") { 24 } else { 32 };
|
|
142
|
+
let spec = hound::WavSpec {
|
|
143
|
+
channels: crate::core::audio::engine::CHANNELS,
|
|
144
|
+
sample_rate: sr,
|
|
145
|
+
bits_per_sample: bits,
|
|
146
|
+
sample_format: hound::SampleFormat::Int,
|
|
147
|
+
};
|
|
148
|
+
|
|
149
|
+
let mut writer = hound::WavWriter::create(output_dir, spec)
|
|
150
|
+
.map_err(|e| format!("Error creating WAV file: {}", e))?;
|
|
151
|
+
|
|
152
|
+
for &s in buffer.iter() {
|
|
153
|
+
let v32 = (s as i32) << (bits - 16);
|
|
154
|
+
writer
|
|
155
|
+
.write_sample(v32)
|
|
156
|
+
.map_err(|e| format!("Error writing sample: {:?}", e))?;
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
writer
|
|
160
|
+
.finalize()
|
|
161
|
+
.map_err(|e| format!("Error finalizing WAV: {:?}", e))?;
|
|
162
|
+
}
|
|
163
|
+
_ => {
|
|
164
|
+
return Err(format!("Unsupported audio format: {}", format_str));
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
Ok(())
|
|
169
|
+
}
|