1use crate::discover_workspace;
2use crate::errors::{Result, SampoError};
3use crate::manifest::{ManifestMetadata, update_manifest_versions};
4use crate::release::{parse_version_string, regenerate_lockfile, restore_prerelease_changesets};
5use crate::types::{
6 PackageInfo, PackageSpecifier, SpecResolution, Workspace, format_ambiguity_options,
7};
8use semver::{BuildMetadata, Prerelease};
9use std::collections::{BTreeMap, BTreeSet};
10use std::fs;
11use std::path::Path;
12
13#[derive(Debug, Clone, PartialEq, Eq)]
15pub struct VersionChange {
16 pub name: String,
17 pub old_version: String,
18 pub new_version: String,
19}
20
21pub fn enter_prerelease(
23 root: &Path,
24 packages: &[String],
25 label: &str,
26) -> Result<Vec<VersionChange>> {
27 let workspace = discover_workspace(root)?;
28 let targets = resolve_targets(&workspace, packages)?;
29 let prerelease = validate_label(label)?;
30
31 let (changes, new_versions) = plan_enter_updates(&targets, &prerelease)?;
32 if new_versions.is_empty() {
33 return Ok(Vec::new());
34 }
35
36 apply_version_updates(&workspace, &new_versions)?;
37 Ok(changes)
38}
39
40pub fn exit_prerelease(root: &Path, packages: &[String]) -> Result<Vec<VersionChange>> {
42 let workspace = discover_workspace(root)?;
43 let targets = resolve_targets(&workspace, packages)?;
44
45 let (changes, new_versions) = plan_exit_updates(&targets)?;
46 if new_versions.is_empty() {
47 return Ok(Vec::new());
48 }
49
50 apply_version_updates(&workspace, &new_versions)?;
51 Ok(changes)
52}
53
54pub fn restore_preserved_changesets(root: &Path) -> Result<usize> {
60 let prerelease_dir = root.join(".sampo").join("prerelease");
61 if !prerelease_dir.exists() {
62 return Ok(0);
63 }
64
65 let changesets_dir = root.join(".sampo").join("changesets");
66 let mut preserved = 0usize;
67
68 for entry in fs::read_dir(&prerelease_dir)? {
69 let entry = entry?;
70 let path = entry.path();
71 if !path.is_file() {
72 continue;
73 }
74 if path.extension().and_then(|ext| ext.to_str()) != Some("md") {
75 continue;
76 }
77 preserved += 1;
78 }
79
80 if preserved == 0 {
81 return Ok(0);
82 }
83
84 restore_prerelease_changesets(&prerelease_dir, &changesets_dir)?;
85 Ok(preserved)
86}
87
88fn resolve_targets<'a>(
89 workspace: &'a Workspace,
90 packages: &[String],
91) -> Result<Vec<&'a PackageInfo>> {
92 if packages.is_empty() {
93 return Err(SampoError::Prerelease(
94 "At least one package must be specified.".to_string(),
95 ));
96 }
97
98 let mut seen: BTreeSet<String> = BTreeSet::new();
99 let mut targets = Vec::new();
100
101 for raw in packages {
102 let spec = PackageSpecifier::parse(raw).map_err(|reason| {
103 SampoError::Prerelease(format!("Invalid package reference '{}': {}", raw, reason))
104 })?;
105
106 let info = match workspace.resolve_specifier(&spec) {
107 SpecResolution::Match(info) => info,
108 SpecResolution::NotFound { query } => {
109 return Err(SampoError::NotFound(format!(
110 "Package '{}' not found in workspace",
111 query.display()
112 )));
113 }
114 SpecResolution::Ambiguous { query, matches } => {
115 let options = format_ambiguity_options(&matches);
116 return Err(SampoError::Prerelease(format!(
117 "Package '{}' is ambiguous. Disambiguate using one of: {}.",
118 query.base_name(),
119 options
120 )));
121 }
122 };
123
124 let identifier = info.canonical_identifier().to_string();
125 if seen.insert(identifier) {
126 targets.push(info);
127 }
128 }
129
130 targets.sort_by(|a, b| a.name.cmp(&b.name));
131 Ok(targets)
132}
133
134fn validate_label(label: &str) -> Result<Prerelease> {
135 let trimmed = label.trim();
136 if trimmed.is_empty() {
137 return Err(SampoError::Prerelease(
138 "Pre-release label cannot be empty.".to_string(),
139 ));
140 }
141
142 let has_non_numeric = trimmed
143 .split('.')
144 .any(|segment| segment.chars().any(|ch| !ch.is_ascii_digit()));
145 if !has_non_numeric {
146 return Err(SampoError::Prerelease(
147 "Pre-release label must contain at least one non-numeric identifier.".to_string(),
148 ));
149 }
150
151 Prerelease::new(trimmed).map_err(|err| {
152 SampoError::Prerelease(format!("Invalid pre-release label '{}': {err}", trimmed))
153 })
154}
155
156fn plan_enter_updates(
157 targets: &[&PackageInfo],
158 prerelease: &Prerelease,
159) -> Result<(Vec<VersionChange>, BTreeMap<String, String>)> {
160 let mut changes = Vec::new();
161 let mut new_versions: BTreeMap<String, String> = BTreeMap::new();
162
163 for info in targets {
164 let version = parse_version_string(&info.version).map_err(|err| {
165 SampoError::Prerelease(format!(
166 "Invalid semantic version for package '{}': {}",
167 info.name, err
168 ))
169 })?;
170
171 let mut base = version.clone();
172 base.pre = Prerelease::EMPTY;
173 base.build = BuildMetadata::EMPTY;
174
175 let mut updated = base.clone();
176 updated.pre = prerelease.clone();
177 let new_version = updated.to_string();
178
179 if new_version == info.version {
180 continue;
181 }
182
183 new_versions.insert(info.name.clone(), new_version.clone());
184 changes.push(VersionChange {
185 name: info.name.clone(),
186 old_version: info.version.clone(),
187 new_version,
188 });
189 }
190
191 Ok((changes, new_versions))
192}
193
194fn plan_exit_updates(
195 targets: &[&PackageInfo],
196) -> Result<(Vec<VersionChange>, BTreeMap<String, String>)> {
197 let mut changes = Vec::new();
198 let mut new_versions: BTreeMap<String, String> = BTreeMap::new();
199
200 for info in targets {
201 let version = parse_version_string(&info.version).map_err(|err| {
202 SampoError::Prerelease(format!(
203 "Invalid semantic version for package '{}': {}",
204 info.name, err
205 ))
206 })?;
207
208 if version.pre.is_empty() {
209 continue;
210 }
211
212 let mut stable = version.clone();
213 stable.pre = Prerelease::EMPTY;
214 stable.build = BuildMetadata::EMPTY;
215 let new_version = stable.to_string();
216
217 if new_version == info.version {
218 continue;
219 }
220
221 new_versions.insert(info.name.clone(), new_version.clone());
222 changes.push(VersionChange {
223 name: info.name.clone(),
224 old_version: info.version.clone(),
225 new_version,
226 });
227 }
228
229 Ok((changes, new_versions))
230}
231
232fn apply_version_updates(
233 workspace: &Workspace,
234 new_versions: &BTreeMap<String, String>,
235) -> Result<()> {
236 let manifest_metadata = ManifestMetadata::load(workspace).map_err(|err| match err {
237 SampoError::Release(msg) => SampoError::Prerelease(msg),
238 other => other,
239 })?;
240
241 let adapter = crate::adapters::PackageAdapter::Cargo;
242 for info in &workspace.members {
243 let manifest_path = adapter.manifest_path(&info.path);
244 let original = fs::read_to_string(&manifest_path)?;
245 let new_pkg_version = new_versions.get(&info.name).map(|s| s.as_str());
246 let (updated, _deps) = update_manifest_versions(
247 &manifest_path,
248 &original,
249 new_pkg_version,
250 new_versions,
251 Some(&manifest_metadata),
252 )?;
253
254 if updated != original {
255 fs::write(&manifest_path, updated)?;
256 }
257 }
258
259 if workspace.root.join("Cargo.lock").exists() {
260 regenerate_lockfile(&workspace.root).map_err(|err| match err {
261 SampoError::Release(msg) => SampoError::Prerelease(msg),
262 other => other,
263 })?;
264 }
265
266 Ok(())
267}
268
269#[cfg(test)]
270mod tests {
271 use super::*;
272 use tempfile::tempdir;
273
274 fn init_workspace() -> tempfile::TempDir {
275 let temp = tempdir().unwrap();
276 let root = temp.path();
277
278 fs::create_dir_all(root.join("crates/foo")).unwrap();
279 fs::create_dir_all(root.join("crates/bar")).unwrap();
280
281 fs::write(
282 root.join("Cargo.toml"),
283 "[workspace]\nmembers=[\"crates/*\"]\n",
284 )
285 .unwrap();
286
287 write_manifest(&root.join("crates/foo"), "foo", "0.1.0");
288 write_manifest(&root.join("crates/bar"), "bar", "0.1.0");
289
290 temp
291 }
292
293 fn write_manifest(path: &Path, name: &str, version: &str) {
294 fs::create_dir_all(path.join("src")).unwrap();
295 fs::write(
296 path.join("Cargo.toml"),
297 format!("[package]\nname = \"{name}\"\nversion = \"{version}\"\n"),
298 )
299 .unwrap();
300 fs::write(path.join("src/lib.rs"), "pub fn __sampo_test_marker() {}\n").unwrap();
301 }
302
303 fn append_dependency(path: &Path, dep: &str, dep_version: &str) {
304 let manifest_path = path.join("Cargo.toml");
305 let current = fs::read_to_string(&manifest_path).unwrap();
306 fs::write(
307 &manifest_path,
308 format!(
309 "{}\n[dependencies]\n{dep} = {{ path = \"../{dep}\", version = \"{dep_version}\" }}\n",
310 current.trim_end()
311 ),
312 )
313 .unwrap();
314 }
315
316 #[test]
317 fn enter_sets_prerelease_label_and_updates_dependents() {
318 let temp = init_workspace();
319 let root = temp.path();
320
321 write_manifest(&root.join("crates/foo"), "foo", "1.2.3");
322 write_manifest(&root.join("crates/bar"), "bar", "0.1.0");
323 append_dependency(&root.join("crates/bar"), "foo", "1.2.3");
324
325 let updates = enter_prerelease(root, &[String::from("foo")], "alpha").unwrap();
326 assert_eq!(
327 updates,
328 vec![VersionChange {
329 name: "foo".to_string(),
330 old_version: "1.2.3".to_string(),
331 new_version: "1.2.3-alpha".to_string(),
332 }]
333 );
334
335 let foo_manifest = fs::read_to_string(root.join("crates/foo/Cargo.toml")).unwrap();
336 assert!(
337 foo_manifest.contains("version = \"1.2.3-alpha\"")
338 || foo_manifest.contains("version=\"1.2.3-alpha\"")
339 );
340
341 let bar_manifest = fs::read_to_string(root.join("crates/bar/Cargo.toml")).unwrap();
342 assert!(
343 bar_manifest.contains("version = \"1.2.3-alpha\"")
344 || bar_manifest.contains("version=\"1.2.3-alpha\"")
345 );
346 }
347
348 #[test]
349 fn enter_switches_between_labels() {
350 let temp = init_workspace();
351 let root = temp.path();
352
353 write_manifest(&root.join("crates/foo"), "foo", "1.0.0-beta.3");
354
355 let updates = enter_prerelease(root, &[String::from("foo")], "alpha").unwrap();
356 assert_eq!(
357 updates,
358 vec![VersionChange {
359 name: "foo".to_string(),
360 old_version: "1.0.0-beta.3".to_string(),
361 new_version: "1.0.0-alpha".to_string(),
362 }]
363 );
364
365 let foo_manifest = fs::read_to_string(root.join("crates/foo/Cargo.toml")).unwrap();
366 assert!(foo_manifest.contains("1.0.0-alpha"));
367 }
368
369 #[test]
370 fn enter_rejects_numeric_only_label() {
371 let temp = init_workspace();
372 let root = temp.path();
373
374 write_manifest(&root.join("crates/foo"), "foo", "0.1.0");
375
376 let err = enter_prerelease(root, &[String::from("foo")], "123").unwrap_err();
377 match err {
378 SampoError::Prerelease(msg) => {
379 assert!(msg.contains("non-numeric"));
380 }
381 other => panic!("unexpected error: {other:?}"),
382 }
383 }
384
385 #[test]
386 fn exit_clears_prerelease_and_updates_dependents() {
387 let temp = init_workspace();
388 let root = temp.path();
389
390 write_manifest(&root.join("crates/foo"), "foo", "2.3.4-alpha.5");
391 write_manifest(&root.join("crates/bar"), "bar", "0.2.0");
392 append_dependency(&root.join("crates/bar"), "foo", "2.3.4-alpha.5");
393
394 let updates = exit_prerelease(root, &[String::from("foo")]).unwrap();
395 assert_eq!(
396 updates,
397 vec![VersionChange {
398 name: "foo".to_string(),
399 old_version: "2.3.4-alpha.5".to_string(),
400 new_version: "2.3.4".to_string(),
401 }]
402 );
403
404 let foo_manifest = fs::read_to_string(root.join("crates/foo/Cargo.toml")).unwrap();
405 assert!(
406 foo_manifest.contains("version = \"2.3.4\"")
407 || foo_manifest.contains("version=\"2.3.4\"")
408 );
409
410 let bar_manifest = fs::read_to_string(root.join("crates/bar/Cargo.toml")).unwrap();
411 assert!(
412 bar_manifest.contains("version = \"2.3.4\"")
413 || bar_manifest.contains("version=\"2.3.4\"")
414 );
415 }
416
417 #[test]
418 fn restore_preserved_changesets_moves_files() {
419 let temp = init_workspace();
420 let root = temp.path();
421
422 let prerelease_dir = root.join(".sampo/prerelease");
423 fs::create_dir_all(&prerelease_dir).unwrap();
424 fs::write(prerelease_dir.join("change.md"), "---\nfoo: minor\n---\n").unwrap();
425
426 let restored = restore_preserved_changesets(root).unwrap();
427 assert_eq!(restored, 1);
428
429 let changesets_dir = root.join(".sampo/changesets");
430 let restored_entries = fs::read_dir(&changesets_dir)
431 .unwrap()
432 .map(|entry| entry.unwrap().path())
433 .collect::<Vec<_>>();
434 assert_eq!(restored_entries.len(), 1);
435 assert!(
436 restored_entries[0]
437 .file_name()
438 .unwrap()
439 .to_string_lossy()
440 .starts_with("change")
441 );
442
443 let remaining = fs::read_dir(&prerelease_dir).unwrap().collect::<Vec<_>>();
444 assert!(remaining.is_empty());
445 }
446}