@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.
Files changed (207) hide show
  1. package/.devalang +9 -10
  2. package/Cargo.toml +84 -80
  3. package/README.md +10 -7
  4. package/docs/CHANGELOG.md +83 -0
  5. package/docs/ROADMAP.md +6 -2
  6. package/docs/TODO.md +3 -14
  7. package/examples/bus.deva +10 -0
  8. package/examples/chain.deva +19 -0
  9. package/examples/effect.deva +2 -0
  10. package/examples/filter.deva +11 -0
  11. package/examples/lfo.deva +9 -0
  12. package/examples/plugin.deva +10 -10
  13. package/examples/routing.deva +23 -0
  14. package/examples/synth.deva +11 -1
  15. package/examples/synth_types.deva +17 -0
  16. package/out-tsc/bin/project-version.json +6 -0
  17. package/out-tsc/core/functions/index.d.ts +5 -0
  18. package/out-tsc/core/functions/index.js +11 -0
  19. package/out-tsc/pkg/devalang_core.d.ts +2 -0
  20. package/out-tsc/pkg/devalang_core.js +17 -2
  21. package/out-tsc/pkg/devalang_core_bg.wasm.d.ts +1 -0
  22. package/out-tsc/scripts/version/copy-to-binary.d.ts +1 -0
  23. package/out-tsc/scripts/version/copy-to-binary.js +79 -0
  24. package/package.json +23 -10
  25. package/project-version.json +3 -3
  26. package/rust/bindings/Cargo.toml +9 -0
  27. package/rust/bindings/src/lib.rs +86 -0
  28. package/rust/cli/addon/commands.rs +35 -0
  29. package/rust/cli/addon/download.rs +234 -0
  30. package/rust/cli/addon/install.rs +33 -0
  31. package/rust/cli/addon/list.rs +224 -0
  32. package/rust/cli/addon/metadata.rs +124 -0
  33. package/rust/cli/addon/mod.rs +8 -0
  34. package/rust/cli/addon/remove.rs +271 -0
  35. package/rust/cli/addon/update.rs +305 -0
  36. package/rust/cli/{install/addon.rs → addon/utils.rs} +34 -43
  37. package/rust/cli/build/commands.rs +153 -103
  38. package/rust/cli/build/mod.rs +2 -2
  39. package/rust/cli/build/process.rs +165 -146
  40. package/rust/cli/check/mod.rs +208 -208
  41. package/rust/cli/discover/commands.rs +53 -31
  42. package/rust/cli/discover/config.rs +2 -4
  43. package/rust/cli/discover/install.rs +139 -28
  44. package/rust/cli/discover/metadata.rs +3 -3
  45. package/rust/cli/login/commands.rs +124 -124
  46. package/rust/cli/me/commands.rs +52 -0
  47. package/rust/cli/me/mod.rs +1 -0
  48. package/rust/cli/mod.rs +2 -2
  49. package/rust/cli/parser.rs +76 -70
  50. package/rust/cli/play/commands.rs +375 -324
  51. package/rust/cli/play/mod.rs +5 -5
  52. package/rust/cli/play/process.rs +159 -150
  53. package/rust/cli/play/realtime.rs +91 -91
  54. package/rust/cli/telemetry/commands.rs +22 -22
  55. package/rust/cli/telemetry/event_creator.rs +80 -80
  56. package/rust/cli/telemetry/mod.rs +3 -3
  57. package/rust/cli/telemetry/send.rs +51 -51
  58. package/rust/cli/template/commands.rs +69 -69
  59. package/rust/config/driver.rs +112 -103
  60. package/rust/config/mod.rs +3 -3
  61. package/rust/config/ops.rs +26 -26
  62. package/rust/config/settings.rs +101 -101
  63. package/rust/core/audio/engine/driver.rs +237 -0
  64. package/rust/core/audio/engine/export.rs +169 -0
  65. package/rust/core/audio/engine/helpers.rs +178 -170
  66. package/rust/core/audio/engine/mod.rs +56 -7
  67. package/rust/core/audio/engine/notes/dsp.rs +88 -0
  68. package/rust/core/audio/engine/notes/mod.rs +53 -0
  69. package/rust/core/audio/engine/notes/params.rs +294 -0
  70. package/rust/core/audio/engine/sample/insert.rs +300 -0
  71. package/rust/core/audio/engine/sample/mod.rs +40 -0
  72. package/rust/core/audio/engine/sample/padding.rs +170 -0
  73. package/rust/core/audio/evaluator/condition.rs +61 -0
  74. package/rust/core/audio/evaluator/mod.rs +9 -0
  75. package/rust/core/audio/{evaluator.rs → evaluator/numeric.rs} +152 -310
  76. package/rust/core/audio/evaluator/rhs.rs +16 -0
  77. package/rust/core/audio/evaluator/string_expr.rs +94 -0
  78. package/rust/core/audio/interpreter/driver.rs +574 -542
  79. package/rust/core/audio/interpreter/mod.rs +2 -14
  80. package/rust/core/audio/interpreter/statements/arrow_call/interprete.rs +179 -0
  81. package/rust/core/audio/interpreter/statements/arrow_call/methods/chord.rs +398 -0
  82. package/rust/core/audio/interpreter/statements/arrow_call/methods/effects.rs +323 -0
  83. package/rust/core/audio/interpreter/statements/arrow_call/methods/mod.rs +3 -0
  84. package/rust/core/audio/interpreter/statements/arrow_call/methods/note.rs +371 -0
  85. package/rust/core/audio/interpreter/statements/arrow_call/mod.rs +3 -0
  86. package/rust/core/audio/interpreter/statements/arrow_call/types/arp.rs +192 -0
  87. package/rust/core/audio/interpreter/statements/arrow_call/types/mod.rs +24 -0
  88. package/rust/core/audio/interpreter/statements/arrow_call/types/pad.rs +116 -0
  89. package/rust/core/audio/interpreter/statements/arrow_call/types/pluck.rs +97 -0
  90. package/rust/core/audio/interpreter/statements/arrow_call/types/sub.rs +100 -0
  91. package/rust/core/audio/interpreter/{automate.rs → statements/automate.rs} +2 -4
  92. package/rust/core/audio/interpreter/{call.rs → statements/call.rs} +36 -5
  93. package/rust/core/audio/interpreter/{condition.rs → statements/condition.rs} +72 -71
  94. package/rust/core/audio/interpreter/{function.rs → statements/function.rs} +24 -26
  95. package/rust/core/audio/interpreter/{let_.rs → statements/let_.rs} +36 -38
  96. package/rust/core/audio/interpreter/{load.rs → statements/load.rs} +17 -19
  97. package/rust/core/audio/interpreter/{loop_.rs → statements/loop_.rs} +115 -114
  98. package/rust/core/audio/interpreter/statements/mod.rs +12 -0
  99. package/rust/core/audio/interpreter/{sleep.rs → statements/sleep.rs} +28 -28
  100. package/rust/core/audio/interpreter/{spawn.rs → statements/spawn.rs} +54 -4
  101. package/rust/core/audio/interpreter/{tempo.rs → statements/tempo.rs} +40 -40
  102. package/rust/core/audio/interpreter/{trigger.rs → statements/trigger.rs} +242 -239
  103. package/rust/core/audio/loader/trigger.rs +98 -97
  104. package/rust/core/audio/mod.rs +6 -7
  105. package/rust/core/audio/special/easing.rs +189 -189
  106. package/rust/core/audio/special/env.rs +45 -45
  107. package/rust/core/audio/special/math.rs +134 -134
  108. package/rust/core/audio/special/modulator.rs +143 -143
  109. package/rust/core/builder/mod.rs +129 -86
  110. package/rust/core/debugger/{module.rs → logs.rs} +52 -55
  111. package/rust/core/debugger/mod.rs +30 -30
  112. package/rust/core/debugger/store.rs +38 -40
  113. package/rust/core/error/mod.rs +269 -269
  114. package/rust/core/lexer/driver.rs +2 -4
  115. package/rust/core/mod.rs +9 -10
  116. package/rust/core/parser/driver/block.rs +111 -0
  117. package/rust/core/parser/driver/cursor.rs +82 -0
  118. package/rust/core/parser/driver/driver_impl.rs +159 -0
  119. package/rust/core/parser/driver/mod.rs +6 -0
  120. package/rust/core/parser/driver/parse_array.rs +120 -0
  121. package/rust/core/parser/driver/parse_map.rs +247 -0
  122. package/rust/core/parser/driver/parser.rs +160 -0
  123. package/rust/core/parser/handler/arrow_call.rs +90 -15
  124. package/rust/core/parser/handler/at.rs +279 -279
  125. package/rust/core/parser/handler/bank.rs +104 -104
  126. package/rust/core/parser/handler/condition.rs +83 -83
  127. package/rust/core/parser/handler/dot.rs +148 -148
  128. package/rust/core/parser/handler/identifier/automate.rs +254 -254
  129. package/rust/core/parser/handler/identifier/call.rs +91 -91
  130. package/rust/core/parser/handler/identifier/emit.rs +70 -70
  131. package/rust/core/parser/handler/identifier/function.rs +113 -113
  132. package/rust/core/parser/handler/identifier/group.rs +89 -89
  133. package/rust/core/parser/handler/identifier/let_.rs +173 -173
  134. package/rust/core/parser/handler/identifier/mod.rs +55 -55
  135. package/rust/core/parser/handler/identifier/on.rs +107 -107
  136. package/rust/core/parser/handler/identifier/print.rs +49 -49
  137. package/rust/core/parser/handler/identifier/sleep.rs +96 -43
  138. package/rust/core/parser/handler/identifier/spawn.rs +91 -91
  139. package/rust/core/parser/handler/identifier/synth.rs +39 -3
  140. package/rust/core/parser/handler/loop_.rs +194 -194
  141. package/rust/core/parser/handler/pattern.rs +25 -2
  142. package/rust/core/parser/handler/tempo.rs +105 -57
  143. package/rust/core/parser/statement.rs +10 -11
  144. package/rust/core/plugin/loader.rs +137 -137
  145. package/rust/core/plugin/runner/mod.rs +11 -0
  146. package/rust/core/plugin/{runner.rs → runner/non_wasm.rs} +206 -72
  147. package/rust/core/plugin/runner/wasm32.rs +44 -0
  148. package/rust/core/preprocessor/loader/inject.rs +313 -0
  149. package/rust/core/preprocessor/loader/loader_helpers.rs +110 -0
  150. package/rust/core/preprocessor/loader/mod.rs +235 -0
  151. package/rust/core/preprocessor/module.rs +55 -60
  152. package/rust/core/preprocessor/{processor.rs → processor/handlers.rs} +107 -114
  153. package/rust/core/preprocessor/processor/mod.rs +1 -0
  154. package/rust/core/preprocessor/resolver/function.rs +69 -69
  155. package/rust/core/preprocessor/resolver/group.rs +122 -94
  156. package/rust/core/preprocessor/resolver/pattern.rs +14 -2
  157. package/rust/core/store/global.rs +57 -61
  158. package/rust/core/store/mod.rs +1 -5
  159. package/rust/lib.rs +323 -308
  160. package/rust/macros/Cargo.toml +14 -0
  161. package/rust/macros/src/lib.rs +52 -0
  162. package/rust/main.rs +336 -143
  163. package/rust/types/Cargo.toml +1 -1
  164. package/rust/types/src/addons.rs +57 -55
  165. package/rust/types/src/config.rs +82 -74
  166. package/rust/types/src/lib.rs +15 -12
  167. package/rust/types/src/plugin.rs +20 -0
  168. package/rust/types/src/store.rs +139 -0
  169. package/rust/types/src/telemetry.rs +85 -85
  170. package/rust/utils/Cargo.toml +5 -2
  171. package/rust/utils/src/file.rs +477 -94
  172. package/rust/utils/src/first_usage.rs +97 -97
  173. package/rust/utils/src/lib.rs +9 -9
  174. package/rust/utils/src/logger.rs +200 -200
  175. package/rust/utils/src/path.rs +158 -88
  176. package/rust/utils/src/signature.rs +41 -41
  177. package/rust/utils/src/spinner.rs +20 -20
  178. package/rust/utils/src/version.rs +58 -27
  179. package/rust/utils/src/watcher.rs +46 -46
  180. package/rust/web/api.rs +5 -5
  181. package/rust/web/auth.rs +5 -0
  182. package/rust/web/cdn.rs +34 -34
  183. package/rust/web/forge.rs +5 -0
  184. package/rust/web/mod.rs +2 -0
  185. package/tests/integration.rs +21 -21
  186. package/typescript/core/functions/index.ts +11 -0
  187. package/typescript/pkg/devalang_core.ts +20 -4
  188. package/typescript/scripts/version/copy-to-binary.ts +82 -0
  189. package/rust/cli/bank/api.rs +0 -122
  190. package/rust/cli/bank/commands.rs +0 -275
  191. package/rust/cli/bank/mod.rs +0 -29
  192. package/rust/cli/install/bank.rs +0 -53
  193. package/rust/cli/install/commands.rs +0 -35
  194. package/rust/cli/install/mod.rs +0 -4
  195. package/rust/cli/install/plugin.rs +0 -61
  196. package/rust/core/audio/engine/sample.rs +0 -366
  197. package/rust/core/audio/engine/synth.rs +0 -325
  198. package/rust/core/audio/interpreter/arrow_call.rs +0 -311
  199. package/rust/core/audio/renderer.rs +0 -54
  200. package/rust/core/parser/driver.rs +0 -584
  201. package/rust/core/preprocessor/loader.rs +0 -637
  202. package/rust/core/store/export.rs +0 -28
  203. package/rust/core/store/function.rs +0 -40
  204. package/rust/core/store/import.rs +0 -28
  205. package/rust/core/store/variable.rs +0 -51
  206. package/rust/core/utils/mod.rs +0 -1
  207. package/rust/core/utils/path.rs +0 -37
@@ -1,94 +1,477 @@
1
- use include_dir::{Dir, DirEntry};
2
- use std::{fs, path::Path};
3
-
4
- pub fn copy_dir_recursive(dir: &Dir, target_root: &Path, base_path: &Path) {
5
- for entry in dir.entries() {
6
- match entry {
7
- DirEntry::Dir(subdir) => {
8
- copy_dir_recursive(subdir, target_root, base_path);
9
- }
10
- DirEntry::File(file) => {
11
- // Compute the destination path relative to the provided base.
12
- let rel_path = match file.path().strip_prefix(base_path) {
13
- Ok(p) => p.to_owned(),
14
- Err(_) => {
15
- eprintln!(
16
- "Warning: failed to compute relative path for {:?}, skipping",
17
- file.path()
18
- );
19
- continue;
20
- }
21
- };
22
-
23
- let dest_path = target_root.join(rel_path);
24
-
25
- if let Some(parent) = dest_path.parent() {
26
- if let Err(e) = fs::create_dir_all(parent) {
27
- eprintln!(
28
- "Warning: failed to create directory {}: {}",
29
- parent.display(),
30
- e
31
- );
32
- continue;
33
- }
34
- }
35
-
36
- if let Err(e) = fs::write(&dest_path, file.contents()) {
37
- eprintln!("Warning: failed to write {}: {}", dest_path.display(), e);
38
- continue;
39
- }
40
- }
41
- }
42
- }
43
- }
44
-
45
- pub fn format_file_size(bytes: u64) -> String {
46
- const KB: u64 = 1024;
47
- const MB: u64 = 1024 * 1024;
48
-
49
- if bytes >= MB {
50
- format!("{:.2} Mb", (bytes as f64) / (MB as f64))
51
- } else if bytes >= KB {
52
- format!("{:.2} Kb", (bytes as f64) / (KB as f64))
53
- } else {
54
- format!("{} bytes", bytes)
55
- }
56
- }
57
-
58
- pub fn extract_zip_safely(archive_path: &Path, dest: &Path) -> Result<(), String> {
59
- let file = std::fs::File::open(archive_path)
60
- .map_err(|e| format!("Failed to open archive {}: {}", archive_path.display(), e))?;
61
- let mut archive = zip::ZipArchive::new(file)
62
- .map_err(|e| format!("Failed to read archive {}: {}", archive_path.display(), e))?;
63
-
64
- for i in 0..archive.len() {
65
- let mut file = archive
66
- .by_index(i)
67
- .map_err(|e| format!("Failed to access archive entry {}: {}", i, e))?;
68
-
69
- let enclosed = match file.enclosed_name() {
70
- Some(path) => path.to_owned(),
71
- None => {
72
- continue;
73
- }
74
- };
75
-
76
- let outpath = dest.join(enclosed);
77
-
78
- if file.name().ends_with('/') || file.is_dir() {
79
- std::fs::create_dir_all(&outpath)
80
- .map_err(|e| format!("Failed to create dir {}: {}", outpath.display(), e))?;
81
- } else {
82
- if let Some(p) = outpath.parent() {
83
- std::fs::create_dir_all(p)
84
- .map_err(|e| format!("Failed to create parent {}: {}", p.display(), e))?;
85
- }
86
- let mut outfile = std::fs::File::create(&outpath)
87
- .map_err(|e| format!("Failed to create file {}: {}", outpath.display(), e))?;
88
- std::io::copy(&mut file, &mut outfile)
89
- .map_err(|e| format!("Failed to write file {}: {}", outpath.display(), e))?;
90
- }
91
- }
92
-
93
- Ok(())
94
- }
1
+ use include_dir::{ Dir, DirEntry };
2
+ use std::{ fs, path::Path, io::{ Read, Seek, SeekFrom } };
3
+
4
+ pub fn copy_dir_recursive(dir: &Dir, target_root: &Path, base_path: &Path) {
5
+ for entry in dir.entries() {
6
+ match entry {
7
+ DirEntry::Dir(subdir) => {
8
+ copy_dir_recursive(subdir, target_root, base_path);
9
+ }
10
+ DirEntry::File(file) => {
11
+ // Compute the destination path relative to the provided base.
12
+ let rel_path = match file.path().strip_prefix(base_path) {
13
+ Ok(p) => p.to_owned(),
14
+ Err(_) => {
15
+ eprintln!(
16
+ "Warning: failed to compute relative path for {:?}, skipping",
17
+ file.path()
18
+ );
19
+ continue;
20
+ }
21
+ };
22
+
23
+ let dest_path = target_root.join(rel_path);
24
+
25
+ if let Some(parent) = dest_path.parent() {
26
+ if let Err(e) = fs::create_dir_all(parent) {
27
+ eprintln!(
28
+ "Warning: failed to create directory {}: {}",
29
+ parent.display(),
30
+ e
31
+ );
32
+ continue;
33
+ }
34
+ }
35
+
36
+ if let Err(e) = fs::write(&dest_path, file.contents()) {
37
+ eprintln!("Warning: failed to write {}: {}", dest_path.display(), e);
38
+ continue;
39
+ }
40
+ }
41
+ }
42
+ }
43
+ }
44
+
45
+ pub fn format_file_size(bytes: u64) -> String {
46
+ const KB: u64 = 1024;
47
+ const MB: u64 = 1024 * 1024;
48
+
49
+ if bytes >= MB {
50
+ format!("{:.2} Mb", (bytes as f64) / (MB as f64))
51
+ } else if bytes >= KB {
52
+ format!("{:.2} Kb", (bytes as f64) / (KB as f64))
53
+ } else {
54
+ format!("{} bytes", bytes)
55
+ }
56
+ }
57
+
58
+ pub fn extract_zip_safely(archive_path: &Path, dest: &Path) -> Result<(), String> {
59
+ // Open file and peek magic bytes to determine archive format
60
+ let mut file = std::fs::File
61
+ ::open(archive_path)
62
+ .map_err(|e| format!("Failed to open archive {}: {}", archive_path.display(), e))?;
63
+
64
+ let mut magic = [0u8; 4];
65
+ let n = file
66
+ .read(&mut magic)
67
+ .map_err(|e| format!("Failed to read archive header {}: {}", archive_path.display(), e))?;
68
+ file
69
+ .seek(SeekFrom::Start(0))
70
+ .map_err(|e| format!("Failed to seek archive {}: {}", archive_path.display(), e))?;
71
+
72
+ let is_gzip = n >= 2 && magic[0] == 0x1f && magic[1] == 0x8b;
73
+ let is_zip = n >= 2 && magic[0] == 0x50 && magic[1] == 0x4b;
74
+
75
+ if is_gzip {
76
+ // The gzip may contain a tar archive OR a gzipped zip (PK inside).
77
+ // Decompress to a temporary file, inspect its magic bytes, then choose extractor.
78
+ let mut gz = flate2::read::GzDecoder::new(file);
79
+
80
+ let mut named = tempfile::NamedTempFile
81
+ ::new()
82
+ .map_err(|e| format!("Failed to create temp file: {}", e))?;
83
+
84
+ std::io
85
+ ::copy(&mut gz, &mut named)
86
+ .map_err(|e| format!("Failed to decompress gzip to temp: {}", e))?;
87
+ named
88
+ .as_file_mut()
89
+ .seek(SeekFrom::Start(0))
90
+ .map_err(|e| format!("Failed to seek temp file: {}", e))?;
91
+
92
+ // Read magic of decompressed content
93
+ let mut head = [0u8; 4];
94
+ let n = named
95
+ .as_file_mut()
96
+ .read(&mut head)
97
+ .map_err(|e| format!("Failed to read temp archive header: {}", e))?;
98
+ named
99
+ .as_file_mut()
100
+ .seek(SeekFrom::Start(0))
101
+ .map_err(|e| format!("Failed to rewind temp file: {}", e))?;
102
+
103
+ let inner_is_zip = n >= 2 && head[0] == 0x50 && head[1] == 0x4b;
104
+
105
+ if inner_is_zip {
106
+ let tmp_file = named
107
+ .reopen()
108
+ .map_err(|e| format!("Failed to reopen temp file for zip: {}", e))?;
109
+
110
+ let mut archive = zip::ZipArchive
111
+ ::new(tmp_file)
112
+ .map_err(|e| format!("Failed to read zip archive inside gzip: {}", e))?;
113
+
114
+ for i in 0..archive.len() {
115
+ let mut file = archive
116
+ .by_index(i)
117
+ .map_err(|e| format!("Failed to access zip archive entry {}: {}", i, e))?;
118
+ let enclosed = match file.enclosed_name() {
119
+ Some(p) => p.to_owned(),
120
+ None => {
121
+ continue;
122
+ }
123
+ };
124
+ if
125
+ enclosed.is_absolute() ||
126
+ enclosed.components().any(|c| matches!(c, std::path::Component::ParentDir))
127
+ {
128
+ continue;
129
+ }
130
+ let outpath = dest.join(enclosed);
131
+ if file.name().ends_with('/') || file.is_dir() {
132
+ std::fs
133
+ ::create_dir_all(&outpath)
134
+ .map_err(|e| format!("Failed to create dir {}: {}", outpath.display(), e))?;
135
+ } else {
136
+ if let Some(p) = outpath.parent() {
137
+ std::fs
138
+ ::create_dir_all(p)
139
+ .map_err(|e|
140
+ format!("Failed to create parent {}: {}", p.display(), e)
141
+ )?;
142
+ }
143
+ let mut outfile = std::fs::File
144
+ ::create(&outpath)
145
+ .map_err(|e|
146
+ format!("Failed to create file {}: {}", outpath.display(), e)
147
+ )?;
148
+ std::io
149
+ ::copy(&mut file, &mut outfile)
150
+ .map_err(|e| format!("Failed to write file {}: {}", outpath.display(), e))?;
151
+ }
152
+ }
153
+
154
+ // NamedTempFile will be removed on drop
155
+ return Ok(());
156
+ }
157
+
158
+ // otherwise treat as tar
159
+ let tmp_file = named
160
+ .reopen()
161
+ .map_err(|e| format!("Failed to reopen temp file for tar: {}", e))?;
162
+ let mut archive = tar::Archive::new(tmp_file);
163
+
164
+ for entry in archive
165
+ .entries()
166
+ .map_err(|e| format!("Failed to read tar entries (from gzip): {}", e))? {
167
+ let mut entry = entry.map_err(|e|
168
+ format!("Failed to read tar archive entry (from gzip): {}", e)
169
+ )?;
170
+ let path = match entry.path() {
171
+ Ok(p) => p.into_owned(),
172
+ Err(_) => {
173
+ continue;
174
+ }
175
+ };
176
+ if
177
+ path.is_absolute() ||
178
+ path.components().any(|c| matches!(c, std::path::Component::ParentDir))
179
+ {
180
+ continue;
181
+ }
182
+ let outpath = dest.join(&path);
183
+ if let Some(parent) = outpath.parent() {
184
+ std::fs
185
+ ::create_dir_all(parent)
186
+ .map_err(|e| format!("Failed to create parent {}: {}", parent.display(), e))?;
187
+ }
188
+ entry
189
+ .unpack(&outpath)
190
+ .map_err(|e|
191
+ format!("Failed to unpack tar entry to {}: {}", outpath.display(), e)
192
+ )?;
193
+ }
194
+
195
+ return Ok(());
196
+ }
197
+
198
+ if is_zip {
199
+ // Re-open file for zip
200
+ let file = std::fs::File
201
+ ::open(archive_path)
202
+ .map_err(|e|
203
+ format!("Failed to open archive for zip {}: {}", archive_path.display(), e)
204
+ )?;
205
+
206
+ let mut archive = zip::ZipArchive
207
+ ::new(file)
208
+ .map_err(|e| format!("Failed to read zip archive {}: {}", archive_path.display(), e))?;
209
+
210
+ for i in 0..archive.len() {
211
+ let mut file = archive
212
+ .by_index(i)
213
+ .map_err(|e| format!("Failed to access zip archive entry {}: {}", i, e))?;
214
+
215
+ let enclosed = match file.enclosed_name() {
216
+ Some(p) => p.to_owned(),
217
+ None => {
218
+ continue;
219
+ }
220
+ };
221
+
222
+ if
223
+ enclosed.is_absolute() ||
224
+ enclosed.components().any(|c| matches!(c, std::path::Component::ParentDir))
225
+ {
226
+ continue;
227
+ }
228
+
229
+ let outpath = dest.join(enclosed);
230
+ if file.name().ends_with('/') || file.is_dir() {
231
+ std::fs
232
+ ::create_dir_all(&outpath)
233
+ .map_err(|e| format!("Failed to create dir {}: {}", outpath.display(), e))?;
234
+ } else {
235
+ if let Some(p) = outpath.parent() {
236
+ std::fs
237
+ ::create_dir_all(p)
238
+ .map_err(|e| format!("Failed to create parent {}: {}", p.display(), e))?;
239
+ }
240
+ let mut outfile = std::fs::File
241
+ ::create(&outpath)
242
+ .map_err(|e| format!("Failed to create file {}: {}", outpath.display(), e))?;
243
+ std::io
244
+ ::copy(&mut file, &mut outfile)
245
+ .map_err(|e| format!("Failed to write file {}: {}", outpath.display(), e))?;
246
+ }
247
+ }
248
+
249
+ return Ok(());
250
+ }
251
+
252
+ // Unknown magic: try tar.gz first, then zip as fallback
253
+ // Try tar.gz
254
+ let file = std::fs::File
255
+ ::open(archive_path)
256
+ .map_err(|e| format!("Failed to open archive {}: {}", archive_path.display(), e))?;
257
+ if
258
+ let Ok(()) = (|| -> Result<(), String> {
259
+ let gz = flate2::read::GzDecoder::new(&file);
260
+ let mut archive = tar::Archive::new(gz);
261
+ for entry in archive
262
+ .entries()
263
+ .map_err(|e| format!("Failed to read tar entries: {}", e))? {
264
+ let mut entry = entry.map_err(|e|
265
+ format!("Failed to read tar archive entry: {}", e)
266
+ )?;
267
+ let path = match entry.path() {
268
+ Ok(p) => p.into_owned(),
269
+ Err(_) => {
270
+ continue;
271
+ }
272
+ };
273
+ if
274
+ path.is_absolute() ||
275
+ path.components().any(|c| matches!(c, std::path::Component::ParentDir))
276
+ {
277
+ continue;
278
+ }
279
+ let outpath = dest.join(&path);
280
+ if let Some(parent) = outpath.parent() {
281
+ std::fs
282
+ ::create_dir_all(parent)
283
+ .map_err(|e|
284
+ format!("Failed to create parent {}: {}", parent.display(), e)
285
+ )?;
286
+ }
287
+ entry
288
+ .unpack(&outpath)
289
+ .map_err(|e|
290
+ format!("Failed to unpack tar entry to {}: {}", outpath.display(), e)
291
+ )?;
292
+ }
293
+ Ok(())
294
+ })()
295
+ {
296
+ return Ok(());
297
+ }
298
+
299
+ // Fallback to zip
300
+ let file = std::fs::File
301
+ ::open(archive_path)
302
+ .map_err(|e| format!("Failed to open archive for zip {}: {}", archive_path.display(), e))?;
303
+ let mut archive = zip::ZipArchive
304
+ ::new(file)
305
+ .map_err(|e| format!("Failed to read zip archive {}: {}", archive_path.display(), e))?;
306
+ for i in 0..archive.len() {
307
+ let mut file = archive
308
+ .by_index(i)
309
+ .map_err(|e| format!("Failed to access zip archive entry {}: {}", i, e))?;
310
+ let enclosed = match file.enclosed_name() {
311
+ Some(p) => p.to_owned(),
312
+ None => {
313
+ continue;
314
+ }
315
+ };
316
+ if
317
+ enclosed.is_absolute() ||
318
+ enclosed.components().any(|c| matches!(c, std::path::Component::ParentDir))
319
+ {
320
+ continue;
321
+ }
322
+ let outpath = dest.join(enclosed);
323
+ if file.name().ends_with('/') || file.is_dir() {
324
+ std::fs
325
+ ::create_dir_all(&outpath)
326
+ .map_err(|e| format!("Failed to create dir {}: {}", outpath.display(), e))?;
327
+ } else {
328
+ if let Some(p) = outpath.parent() {
329
+ std::fs
330
+ ::create_dir_all(p)
331
+ .map_err(|e| format!("Failed to create parent {}: {}", p.display(), e))?;
332
+ }
333
+ let mut outfile = std::fs::File
334
+ ::create(&outpath)
335
+ .map_err(|e| format!("Failed to create file {}: {}", outpath.display(), e))?;
336
+ std::io
337
+ ::copy(&mut file, &mut outfile)
338
+ .map_err(|e| format!("Failed to write file {}: {}", outpath.display(), e))?;
339
+ }
340
+ }
341
+
342
+ Ok(())
343
+ }
344
+
345
+ /// Detects addon type by inspecting archive contents without extracting fully.
346
+ /// Returns one of: "bank", "plugin", "preset", "template", or "unknown".
347
+ pub fn detect_addon_type_in_archive(archive_path: &Path) -> Result<String, String> {
348
+ use std::fs::File;
349
+
350
+ let mut file = File::open(archive_path)
351
+ .map_err(|e| format!("Failed to open archive {}: {}", archive_path.display(), e))?;
352
+
353
+ let mut magic = [0u8; 4];
354
+ let n = file
355
+ .read(&mut magic)
356
+ .map_err(|e| format!("Failed to read archive header {}: {}", archive_path.display(), e))?;
357
+ file
358
+ .seek(SeekFrom::Start(0))
359
+ .map_err(|e| format!("Failed to seek archive {}: {}", archive_path.display(), e))?;
360
+
361
+ let is_gzip = n >= 2 && magic[0] == 0x1f && magic[1] == 0x8b;
362
+ let is_zip = n >= 2 && magic[0] == 0x50 && magic[1] == 0x4b;
363
+
364
+ let check_name = |name: &str| {
365
+ let lname = name.to_ascii_lowercase();
366
+ if lname.ends_with("bank.toml") {
367
+ return Some("bank".to_string());
368
+ }
369
+ if lname.ends_with("plugin.toml") {
370
+ return Some("plugin".to_string());
371
+ }
372
+ if lname.ends_with("preset.toml") {
373
+ return Some("preset".to_string());
374
+ }
375
+ if lname.ends_with("template.toml") {
376
+ return Some("template".to_string());
377
+ }
378
+ None
379
+ };
380
+
381
+ if is_gzip {
382
+ // Decompress to temp and inspect inner archive
383
+ let mut gz = flate2::read::GzDecoder::new(file);
384
+ let mut named = tempfile::NamedTempFile::new()
385
+ .map_err(|e| format!("Failed to create temp file: {}", e))?;
386
+ std::io::copy(&mut gz, &mut named)
387
+ .map_err(|e| format!("Failed to decompress gzip to temp: {}", e))?;
388
+ named.as_file_mut().seek(SeekFrom::Start(0)).map_err(|e| format!("Failed to seek temp file: {}", e))?;
389
+
390
+ // Inspect inner magic
391
+ let mut head = [0u8; 4];
392
+ let m = named.as_file_mut().read(&mut head).map_err(|e| format!("Failed to read temp archive header: {}", e))?;
393
+ named.as_file_mut().seek(SeekFrom::Start(0)).map_err(|e| format!("Failed to rewind temp file: {}", e))?;
394
+
395
+ let inner_is_zip = m >= 2 && head[0] == 0x50 && head[1] == 0x4b;
396
+ if inner_is_zip {
397
+ let tmp_file = named.reopen().map_err(|e| format!("Failed to reopen temp file for zip: {}", e))?;
398
+ let mut archive = zip::ZipArchive::new(tmp_file).map_err(|e| format!("Failed to read zip archive inside gzip: {}", e))?;
399
+ for i in 0..archive.len() {
400
+ if let Ok(file) = archive.by_index(i) {
401
+ if let Some(enclosed) = file.enclosed_name() {
402
+ let s = enclosed.to_string_lossy().to_string();
403
+ if let Some(t) = check_name(&s) {
404
+ return Ok(t);
405
+ }
406
+ }
407
+ }
408
+ }
409
+ return Ok("unknown".to_string());
410
+ }
411
+
412
+ // treat as tar
413
+ let tmp_file = named.reopen().map_err(|e| format!("Failed to reopen temp file for tar: {}", e))?;
414
+ let mut archive = tar::Archive::new(tmp_file);
415
+ if let Ok(entries) = archive.entries() {
416
+ for entry in entries.filter_map(|e| e.ok()) {
417
+ if let Ok(path) = entry.path() {
418
+ let s = path.to_string_lossy().to_string();
419
+ if let Some(t) = check_name(&s) {
420
+ return Ok(t);
421
+ }
422
+ }
423
+ }
424
+ }
425
+ return Ok("unknown".to_string());
426
+ }
427
+
428
+ if is_zip {
429
+ let file = std::fs::File::open(archive_path).map_err(|e| format!("Failed to open archive for zip {}: {}", archive_path.display(), e))?;
430
+ let mut archive = zip::ZipArchive::new(file).map_err(|e| format!("Failed to read zip archive {}: {}", archive_path.display(), e))?;
431
+ for i in 0..archive.len() {
432
+ if let Ok(file) = archive.by_index(i) {
433
+ if let Some(enclosed) = file.enclosed_name() {
434
+ let s = enclosed.to_string_lossy().to_string();
435
+ if let Some(t) = check_name(&s) {
436
+ return Ok(t);
437
+ }
438
+ }
439
+ }
440
+ }
441
+ return Ok("unknown".to_string());
442
+ }
443
+
444
+ // fallback: try tar.gz read attempt
445
+ if let Ok(f) = std::fs::File::open(archive_path) {
446
+ let gz = flate2::read::GzDecoder::new(f);
447
+ let mut archive = tar::Archive::new(gz);
448
+ if let Ok(entries) = archive.entries() {
449
+ for entry in entries.filter_map(|e| e.ok()) {
450
+ if let Ok(path) = entry.path() {
451
+ let s = path.to_string_lossy().to_string();
452
+ if let Some(t) = check_name(&s) {
453
+ return Ok(t);
454
+ }
455
+ }
456
+ }
457
+ }
458
+ }
459
+
460
+ Ok("unknown".to_string())
461
+ }
462
+
463
+ /// Removes the temporary folder located at `.deva/tmp` inside the project root if it exists.
464
+ /// Returns Ok(()) even if the folder did not exist. Returns Err(String) on filesystem errors.
465
+ pub fn clear_tmp_folder() -> Result<(), String> {
466
+ match crate::path::ensure_deva_dir() {
467
+ Ok(deva_dir) => {
468
+ let tmp_dir = deva_dir.join("tmp");
469
+ if tmp_dir.exists() {
470
+ std::fs::remove_dir_all(&tmp_dir)
471
+ .map_err(|e| format!("Failed to remove tmp dir '{}': {}", tmp_dir.display(), e))?;
472
+ }
473
+ Ok(())
474
+ }
475
+ Err(e) => Err(e),
476
+ }
477
+ }