@devaloop/devalang 0.0.1-beta.1 → 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/.devalang +9 -10
- package/Cargo.toml +5 -4
- package/README.md +7 -5
- package/docs/CHANGELOG.md +42 -0
- package/docs/ROADMAP.md +5 -1
- package/docs/TODO.md +3 -14
- package/examples/bus.deva +10 -0
- package/examples/effect.deva +2 -0
- package/examples/filter.deva +11 -0
- package/examples/lfo.deva +9 -0
- package/examples/synth.deva +11 -1
- package/examples/synth_types.deva +17 -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 +8 -7
- package/package.json +1 -1
- package/project-version.json +3 -3
- package/rust/cli/bank/api.rs +122 -122
- package/rust/cli/bank/commands.rs +33 -2
- package/rust/cli/bank/mod.rs +29 -29
- package/rust/cli/build/commands.rs +53 -3
- package/rust/cli/build/mod.rs +2 -2
- package/rust/cli/build/process.rs +26 -7
- package/rust/cli/check/mod.rs +2 -2
- package/rust/cli/discover/commands.rs +253 -253
- package/rust/cli/discover/config.rs +111 -111
- package/rust/cli/discover/fs.rs +19 -19
- package/rust/cli/discover/install.rs +103 -103
- package/rust/cli/discover/metadata.rs +48 -48
- package/rust/cli/discover/mod.rs +5 -5
- package/rust/cli/install/addon.rs +118 -118
- package/rust/cli/install/bank.rs +22 -3
- package/rust/cli/install/commands.rs +35 -35
- package/rust/cli/install/mod.rs +4 -4
- package/rust/cli/install/plugin.rs +80 -61
- package/rust/cli/login/commands.rs +124 -124
- package/rust/cli/mod.rs +12 -12
- package/rust/cli/parser.rs +46 -1
- package/rust/cli/play/commands.rs +71 -20
- package/rust/cli/play/mod.rs +5 -5
- package/rust/cli/play/process.rs +14 -5
- 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 +220 -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 +51 -2
- 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.rs → evaluator/numeric.rs} +1 -159
- 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 +55 -23
- package/rust/core/audio/interpreter/mod.rs +1 -13
- 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/{automate.rs → statements/automate.rs} +16 -18
- package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +5 -4
- package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +2 -1
- package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +2 -4
- package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +2 -4
- package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +2 -4
- package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +2 -1
- 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} +3 -2
- package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
- package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +1 -1
- package/rust/core/audio/loader/trigger.rs +2 -1
- package/rust/core/audio/mod.rs +6 -7
- package/rust/core/audio/player.rs +70 -70
- 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/mod.rs +9 -9
- package/rust/core/audio/special/modulator.rs +143 -143
- package/rust/core/builder/mod.rs +45 -2
- package/rust/core/debugger/lexer.rs +27 -27
- package/rust/core/debugger/{module.rs → logs.rs} +3 -6
- package/rust/core/debugger/mod.rs +30 -30
- package/rust/core/debugger/preprocessor.rs +27 -27
- package/rust/core/debugger/store.rs +2 -4
- package/rust/core/error/mod.rs +269 -269
- package/rust/core/lexer/driver.rs +59 -61
- package/rust/core/lexer/handler/arrow.rs +82 -82
- package/rust/core/lexer/handler/at.rs +21 -21
- package/rust/core/lexer/handler/brace.rs +41 -41
- package/rust/core/lexer/handler/colon.rs +21 -21
- package/rust/core/lexer/handler/comment.rs +30 -30
- package/rust/core/lexer/handler/dot.rs +21 -21
- package/rust/core/lexer/handler/driver.rs +337 -337
- package/rust/core/lexer/handler/identifier.rs +47 -47
- package/rust/core/lexer/handler/indent.rs +66 -66
- package/rust/core/lexer/handler/mod.rs +15 -15
- package/rust/core/lexer/handler/newline.rs +23 -23
- package/rust/core/lexer/handler/number.rs +31 -31
- package/rust/core/lexer/handler/operator.rs +46 -46
- package/rust/core/lexer/handler/parenthesis.rs +41 -41
- package/rust/core/lexer/handler/slash.rs +21 -21
- package/rust/core/lexer/handler/string.rs +63 -63
- package/rust/core/lexer/mod.rs +3 -3
- package/rust/core/mod.rs +0 -1
- 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 +28 -4
- 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 +135 -135
- package/rust/core/parser/handler/loop_.rs +194 -194
- package/rust/core/parser/handler/mod.rs +9 -9
- package/rust/core/parser/handler/pattern.rs +1 -1
- package/rust/core/parser/handler/tempo.rs +105 -57
- package/rust/core/parser/statement.rs +10 -11
- package/rust/core/plugin/loader.rs +1 -1
- package/rust/core/plugin/mod.rs +2 -2
- package/rust/core/plugin/runner/mod.rs +11 -0
- package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +297 -347
- 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/module.rs +2 -7
- package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +6 -13
- package/rust/core/preprocessor/processor/mod.rs +1 -0
- package/rust/core/preprocessor/resolver/bank.rs +49 -49
- package/rust/core/preprocessor/resolver/call.rs +124 -124
- package/rust/core/preprocessor/resolver/condition.rs +95 -95
- package/rust/core/preprocessor/resolver/driver.rs +324 -324
- package/rust/core/preprocessor/resolver/function.rs +2 -2
- package/rust/core/preprocessor/resolver/group.rs +46 -18
- package/rust/core/preprocessor/resolver/let_.rs +32 -32
- package/rust/core/preprocessor/resolver/loop_.rs +318 -318
- package/rust/core/preprocessor/resolver/mod.rs +16 -16
- package/rust/core/preprocessor/resolver/pattern.rs +83 -83
- package/rust/core/preprocessor/resolver/spawn.rs +99 -99
- package/rust/core/preprocessor/resolver/synth.rs +54 -54
- package/rust/core/preprocessor/resolver/tempo.rs +48 -48
- package/rust/core/preprocessor/resolver/trigger.rs +116 -116
- package/rust/core/preprocessor/resolver/value.rs +176 -176
- package/rust/core/store/global.rs +2 -6
- package/rust/core/store/mod.rs +1 -5
- package/rust/lib.rs +18 -3
- package/rust/main.rs +27 -3
- package/rust/types/Cargo.toml +1 -1
- package/rust/types/src/addons.rs +55 -55
- package/rust/types/src/config.rs +84 -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 +2 -2
- package/rust/utils/src/file.rs +94 -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 +129 -88
- package/rust/utils/src/signature.rs +41 -41
- package/rust/utils/src/spinner.rs +20 -20
- package/rust/utils/src/version.rs +27 -27
- package/rust/utils/src/watcher.rs +46 -46
- package/rust/web/api.rs +5 -5
- package/rust/web/cdn.rs +34 -34
- package/rust/web/mod.rs +3 -3
- package/tests/integration.rs +21 -21
- package/typescript/core/functions/index.ts +11 -0
- package/typescript/pkg/devalang_core.ts +20 -4
- 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/utils/src/logger.rs
CHANGED
|
@@ -1,200 +1,200 @@
|
|
|
1
|
-
#[cfg(feature = "cli")]
|
|
2
|
-
use crossterm::style::{Attribute, Color, ResetColor, SetAttribute, SetForegroundColor};
|
|
3
|
-
#[cfg(feature = "cli")]
|
|
4
|
-
use std::fmt::Write;
|
|
5
|
-
|
|
6
|
-
#[derive(Debug, Clone, PartialEq)]
|
|
7
|
-
pub enum LogLevel {
|
|
8
|
-
Success,
|
|
9
|
-
Error,
|
|
10
|
-
Info,
|
|
11
|
-
Print,
|
|
12
|
-
Warning,
|
|
13
|
-
Watcher,
|
|
14
|
-
Debug,
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
#[derive(Debug, Clone)]
|
|
18
|
-
pub struct Logger;
|
|
19
|
-
|
|
20
|
-
impl Logger {
|
|
21
|
-
pub fn new() -> Self {
|
|
22
|
-
Logger
|
|
23
|
-
}
|
|
24
|
-
|
|
25
|
-
// Some methods are only used in CLI builds; silence dead_code warnings for non-cli.
|
|
26
|
-
#[cfg(not(feature = "cli"))]
|
|
27
|
-
#[allow(dead_code)]
|
|
28
|
-
fn __wasm_only_allow_dead_code(&self) {}
|
|
29
|
-
|
|
30
|
-
// Additionally allow dead_code for the CLI-only formatting helpers when building without cli
|
|
31
|
-
#[cfg(not(feature = "cli"))]
|
|
32
|
-
#[allow(dead_code)]
|
|
33
|
-
fn __wasm_only_format_helpers(&self) {}
|
|
34
|
-
|
|
35
|
-
#[cfg(feature = "cli")]
|
|
36
|
-
pub fn log_message(&self, level: LogLevel, message: &str) {
|
|
37
|
-
let formatted_status = self.format_status(level);
|
|
38
|
-
println!(
|
|
39
|
-
"🦊 {} {} {}",
|
|
40
|
-
self.language_signature(),
|
|
41
|
-
formatted_status,
|
|
42
|
-
message
|
|
43
|
-
);
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
#[cfg(not(feature = "cli"))]
|
|
47
|
-
pub fn log_message(&self, _level: LogLevel, _message: &str) {
|
|
48
|
-
// no-op for WASM
|
|
49
|
-
}
|
|
50
|
-
|
|
51
|
-
#[cfg(feature = "cli")]
|
|
52
|
-
pub fn log_message_with_trace(&self, level: LogLevel, message: &str, trace: Vec<&str>) {
|
|
53
|
-
let formatted_status = self.format_status(level);
|
|
54
|
-
println!(
|
|
55
|
-
"🦊 {} {} {}",
|
|
56
|
-
self.language_signature(),
|
|
57
|
-
formatted_status,
|
|
58
|
-
message
|
|
59
|
-
);
|
|
60
|
-
for t in trace {
|
|
61
|
-
println!(" ↳ {}", t);
|
|
62
|
-
}
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
#[cfg(not(feature = "cli"))]
|
|
66
|
-
pub fn log_message_with_trace(&self, _level: LogLevel, _message: &str, _trace: Vec<&str>) {
|
|
67
|
-
// no-op for WASM
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
#[cfg(feature = "cli")]
|
|
71
|
-
pub fn log_error_with_stacktrace(&self, message: &str, stacktrace: &str) {
|
|
72
|
-
let formatted_status = self.format_status(LogLevel::Error);
|
|
73
|
-
println!(
|
|
74
|
-
"🦊 {} {} {}",
|
|
75
|
-
self.language_signature(),
|
|
76
|
-
formatted_status,
|
|
77
|
-
message
|
|
78
|
-
);
|
|
79
|
-
println!(" ↳ {}", stacktrace);
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
#[cfg(not(feature = "cli"))]
|
|
83
|
-
pub fn log_error_with_stacktrace(&self, _message: &str, _stacktrace: &str) {
|
|
84
|
-
// no-op for WASM
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
#[cfg(feature = "cli")]
|
|
88
|
-
#[allow(dead_code)]
|
|
89
|
-
#[cfg_attr(not(feature = "cli"), allow(dead_code))]
|
|
90
|
-
fn language_signature(&self) -> String {
|
|
91
|
-
let mut s = String::new();
|
|
92
|
-
|
|
93
|
-
write!(&mut s, "{}", SetForegroundColor(Color::Grey)).unwrap();
|
|
94
|
-
s.push('[');
|
|
95
|
-
|
|
96
|
-
write!(
|
|
97
|
-
&mut s,
|
|
98
|
-
"{}",
|
|
99
|
-
SetForegroundColor(Color::Rgb {
|
|
100
|
-
r: 29,
|
|
101
|
-
g: 211,
|
|
102
|
-
b: 176,
|
|
103
|
-
})
|
|
104
|
-
)
|
|
105
|
-
.unwrap();
|
|
106
|
-
write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
|
|
107
|
-
s.push_str("Devalang");
|
|
108
|
-
write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
|
|
109
|
-
|
|
110
|
-
write!(&mut s, "{}", SetForegroundColor(Color::Grey)).unwrap();
|
|
111
|
-
s.push(']');
|
|
112
|
-
write!(&mut s, "{}", ResetColor).unwrap();
|
|
113
|
-
|
|
114
|
-
s
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
#[cfg(not(feature = "cli"))]
|
|
118
|
-
#[allow(dead_code)]
|
|
119
|
-
fn language_signature(&self) -> String {
|
|
120
|
-
"[Devalang]".to_string()
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
#[cfg(feature = "cli")]
|
|
124
|
-
fn format_status(&self, level: LogLevel) -> String {
|
|
125
|
-
let mut s = String::new();
|
|
126
|
-
|
|
127
|
-
let color = match level {
|
|
128
|
-
LogLevel::Success => Color::Rgb {
|
|
129
|
-
r: 76,
|
|
130
|
-
g: 175,
|
|
131
|
-
b: 80,
|
|
132
|
-
},
|
|
133
|
-
LogLevel::Error => Color::Rgb {
|
|
134
|
-
r: 244,
|
|
135
|
-
g: 67,
|
|
136
|
-
b: 54,
|
|
137
|
-
},
|
|
138
|
-
LogLevel::Info => Color::Rgb {
|
|
139
|
-
r: 33,
|
|
140
|
-
g: 150,
|
|
141
|
-
b: 243,
|
|
142
|
-
},
|
|
143
|
-
LogLevel::Warning => Color::Rgb {
|
|
144
|
-
r: 255,
|
|
145
|
-
g: 152,
|
|
146
|
-
b: 0,
|
|
147
|
-
},
|
|
148
|
-
LogLevel::Watcher => Color::Rgb {
|
|
149
|
-
r: 156,
|
|
150
|
-
g: 39,
|
|
151
|
-
b: 176,
|
|
152
|
-
},
|
|
153
|
-
LogLevel::Debug => Color::Rgb {
|
|
154
|
-
r: 103,
|
|
155
|
-
g: 58,
|
|
156
|
-
b: 183,
|
|
157
|
-
},
|
|
158
|
-
LogLevel::Print => Color::Rgb {
|
|
159
|
-
r: 255,
|
|
160
|
-
g: 255,
|
|
161
|
-
b: 255,
|
|
162
|
-
},
|
|
163
|
-
};
|
|
164
|
-
|
|
165
|
-
let status = match level {
|
|
166
|
-
LogLevel::Success => "SUCCESS",
|
|
167
|
-
LogLevel::Error => "ERROR",
|
|
168
|
-
LogLevel::Info => "INFO",
|
|
169
|
-
LogLevel::Warning => "WARNING",
|
|
170
|
-
LogLevel::Watcher => "WATCHER",
|
|
171
|
-
LogLevel::Debug => "DEBUG",
|
|
172
|
-
LogLevel::Print => "PRINT",
|
|
173
|
-
};
|
|
174
|
-
|
|
175
|
-
s.push('[');
|
|
176
|
-
write!(&mut s, "{}", SetForegroundColor(color)).unwrap();
|
|
177
|
-
write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
|
|
178
|
-
s.push_str(status);
|
|
179
|
-
write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
|
|
180
|
-
s.push(']');
|
|
181
|
-
write!(&mut s, "{}", ResetColor).unwrap();
|
|
182
|
-
|
|
183
|
-
s
|
|
184
|
-
}
|
|
185
|
-
|
|
186
|
-
#[cfg(not(feature = "cli"))]
|
|
187
|
-
#[allow(dead_code)]
|
|
188
|
-
fn format_status(&self, level: LogLevel) -> String {
|
|
189
|
-
(match level {
|
|
190
|
-
LogLevel::Success => "[SUCCESS]",
|
|
191
|
-
LogLevel::Error => "[ERROR]",
|
|
192
|
-
LogLevel::Info => "[INFO]",
|
|
193
|
-
LogLevel::Warning => "[WARNING]",
|
|
194
|
-
LogLevel::Watcher => "[WATCHER]",
|
|
195
|
-
LogLevel::Debug => "[DEBUG]",
|
|
196
|
-
LogLevel::Print => "[PRINT]",
|
|
197
|
-
})
|
|
198
|
-
.to_string()
|
|
199
|
-
}
|
|
200
|
-
}
|
|
1
|
+
#[cfg(feature = "cli")]
|
|
2
|
+
use crossterm::style::{Attribute, Color, ResetColor, SetAttribute, SetForegroundColor};
|
|
3
|
+
#[cfg(feature = "cli")]
|
|
4
|
+
use std::fmt::Write;
|
|
5
|
+
|
|
6
|
+
#[derive(Debug, Clone, PartialEq)]
|
|
7
|
+
pub enum LogLevel {
|
|
8
|
+
Success,
|
|
9
|
+
Error,
|
|
10
|
+
Info,
|
|
11
|
+
Print,
|
|
12
|
+
Warning,
|
|
13
|
+
Watcher,
|
|
14
|
+
Debug,
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
#[derive(Debug, Clone)]
|
|
18
|
+
pub struct Logger;
|
|
19
|
+
|
|
20
|
+
impl Logger {
|
|
21
|
+
pub fn new() -> Self {
|
|
22
|
+
Logger
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
// Some methods are only used in CLI builds; silence dead_code warnings for non-cli.
|
|
26
|
+
#[cfg(not(feature = "cli"))]
|
|
27
|
+
#[allow(dead_code)]
|
|
28
|
+
fn __wasm_only_allow_dead_code(&self) {}
|
|
29
|
+
|
|
30
|
+
// Additionally allow dead_code for the CLI-only formatting helpers when building without cli
|
|
31
|
+
#[cfg(not(feature = "cli"))]
|
|
32
|
+
#[allow(dead_code)]
|
|
33
|
+
fn __wasm_only_format_helpers(&self) {}
|
|
34
|
+
|
|
35
|
+
#[cfg(feature = "cli")]
|
|
36
|
+
pub fn log_message(&self, level: LogLevel, message: &str) {
|
|
37
|
+
let formatted_status = self.format_status(level);
|
|
38
|
+
println!(
|
|
39
|
+
"🦊 {} {} {}",
|
|
40
|
+
self.language_signature(),
|
|
41
|
+
formatted_status,
|
|
42
|
+
message
|
|
43
|
+
);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
#[cfg(not(feature = "cli"))]
|
|
47
|
+
pub fn log_message(&self, _level: LogLevel, _message: &str) {
|
|
48
|
+
// no-op for WASM
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
#[cfg(feature = "cli")]
|
|
52
|
+
pub fn log_message_with_trace(&self, level: LogLevel, message: &str, trace: Vec<&str>) {
|
|
53
|
+
let formatted_status = self.format_status(level);
|
|
54
|
+
println!(
|
|
55
|
+
"🦊 {} {} {}",
|
|
56
|
+
self.language_signature(),
|
|
57
|
+
formatted_status,
|
|
58
|
+
message
|
|
59
|
+
);
|
|
60
|
+
for t in trace {
|
|
61
|
+
println!(" ↳ {}", t);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
#[cfg(not(feature = "cli"))]
|
|
66
|
+
pub fn log_message_with_trace(&self, _level: LogLevel, _message: &str, _trace: Vec<&str>) {
|
|
67
|
+
// no-op for WASM
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
#[cfg(feature = "cli")]
|
|
71
|
+
pub fn log_error_with_stacktrace(&self, message: &str, stacktrace: &str) {
|
|
72
|
+
let formatted_status = self.format_status(LogLevel::Error);
|
|
73
|
+
println!(
|
|
74
|
+
"🦊 {} {} {}",
|
|
75
|
+
self.language_signature(),
|
|
76
|
+
formatted_status,
|
|
77
|
+
message
|
|
78
|
+
);
|
|
79
|
+
println!(" ↳ {}", stacktrace);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
#[cfg(not(feature = "cli"))]
|
|
83
|
+
pub fn log_error_with_stacktrace(&self, _message: &str, _stacktrace: &str) {
|
|
84
|
+
// no-op for WASM
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
#[cfg(feature = "cli")]
|
|
88
|
+
#[allow(dead_code)]
|
|
89
|
+
#[cfg_attr(not(feature = "cli"), allow(dead_code))]
|
|
90
|
+
fn language_signature(&self) -> String {
|
|
91
|
+
let mut s = String::new();
|
|
92
|
+
|
|
93
|
+
write!(&mut s, "{}", SetForegroundColor(Color::Grey)).unwrap();
|
|
94
|
+
s.push('[');
|
|
95
|
+
|
|
96
|
+
write!(
|
|
97
|
+
&mut s,
|
|
98
|
+
"{}",
|
|
99
|
+
SetForegroundColor(Color::Rgb {
|
|
100
|
+
r: 29,
|
|
101
|
+
g: 211,
|
|
102
|
+
b: 176,
|
|
103
|
+
})
|
|
104
|
+
)
|
|
105
|
+
.unwrap();
|
|
106
|
+
write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
|
|
107
|
+
s.push_str("Devalang");
|
|
108
|
+
write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
|
|
109
|
+
|
|
110
|
+
write!(&mut s, "{}", SetForegroundColor(Color::Grey)).unwrap();
|
|
111
|
+
s.push(']');
|
|
112
|
+
write!(&mut s, "{}", ResetColor).unwrap();
|
|
113
|
+
|
|
114
|
+
s
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
#[cfg(not(feature = "cli"))]
|
|
118
|
+
#[allow(dead_code)]
|
|
119
|
+
fn language_signature(&self) -> String {
|
|
120
|
+
"[Devalang]".to_string()
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
#[cfg(feature = "cli")]
|
|
124
|
+
fn format_status(&self, level: LogLevel) -> String {
|
|
125
|
+
let mut s = String::new();
|
|
126
|
+
|
|
127
|
+
let color = match level {
|
|
128
|
+
LogLevel::Success => Color::Rgb {
|
|
129
|
+
r: 76,
|
|
130
|
+
g: 175,
|
|
131
|
+
b: 80,
|
|
132
|
+
},
|
|
133
|
+
LogLevel::Error => Color::Rgb {
|
|
134
|
+
r: 244,
|
|
135
|
+
g: 67,
|
|
136
|
+
b: 54,
|
|
137
|
+
},
|
|
138
|
+
LogLevel::Info => Color::Rgb {
|
|
139
|
+
r: 33,
|
|
140
|
+
g: 150,
|
|
141
|
+
b: 243,
|
|
142
|
+
},
|
|
143
|
+
LogLevel::Warning => Color::Rgb {
|
|
144
|
+
r: 255,
|
|
145
|
+
g: 152,
|
|
146
|
+
b: 0,
|
|
147
|
+
},
|
|
148
|
+
LogLevel::Watcher => Color::Rgb {
|
|
149
|
+
r: 156,
|
|
150
|
+
g: 39,
|
|
151
|
+
b: 176,
|
|
152
|
+
},
|
|
153
|
+
LogLevel::Debug => Color::Rgb {
|
|
154
|
+
r: 103,
|
|
155
|
+
g: 58,
|
|
156
|
+
b: 183,
|
|
157
|
+
},
|
|
158
|
+
LogLevel::Print => Color::Rgb {
|
|
159
|
+
r: 255,
|
|
160
|
+
g: 255,
|
|
161
|
+
b: 255,
|
|
162
|
+
},
|
|
163
|
+
};
|
|
164
|
+
|
|
165
|
+
let status = match level {
|
|
166
|
+
LogLevel::Success => "SUCCESS",
|
|
167
|
+
LogLevel::Error => "ERROR",
|
|
168
|
+
LogLevel::Info => "INFO",
|
|
169
|
+
LogLevel::Warning => "WARNING",
|
|
170
|
+
LogLevel::Watcher => "WATCHER",
|
|
171
|
+
LogLevel::Debug => "DEBUG",
|
|
172
|
+
LogLevel::Print => "PRINT",
|
|
173
|
+
};
|
|
174
|
+
|
|
175
|
+
s.push('[');
|
|
176
|
+
write!(&mut s, "{}", SetForegroundColor(color)).unwrap();
|
|
177
|
+
write!(&mut s, "{}", SetAttribute(Attribute::Bold)).unwrap();
|
|
178
|
+
s.push_str(status);
|
|
179
|
+
write!(&mut s, "{}", SetAttribute(Attribute::Reset)).unwrap();
|
|
180
|
+
s.push(']');
|
|
181
|
+
write!(&mut s, "{}", ResetColor).unwrap();
|
|
182
|
+
|
|
183
|
+
s
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
#[cfg(not(feature = "cli"))]
|
|
187
|
+
#[allow(dead_code)]
|
|
188
|
+
fn format_status(&self, level: LogLevel) -> String {
|
|
189
|
+
(match level {
|
|
190
|
+
LogLevel::Success => "[SUCCESS]",
|
|
191
|
+
LogLevel::Error => "[ERROR]",
|
|
192
|
+
LogLevel::Info => "[INFO]",
|
|
193
|
+
LogLevel::Warning => "[WARNING]",
|
|
194
|
+
LogLevel::Watcher => "[WATCHER]",
|
|
195
|
+
LogLevel::Debug => "[DEBUG]",
|
|
196
|
+
LogLevel::Print => "[PRINT]",
|
|
197
|
+
})
|
|
198
|
+
.to_string()
|
|
199
|
+
}
|
|
200
|
+
}
|
package/rust/utils/src/path.rs
CHANGED
|
@@ -1,88 +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
|
-
}
|
|
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
|
+
}
|