sampo_core/
prerelease.rs

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/// Represents a version change applied to a package manifest.
14#[derive(Debug, Clone, PartialEq, Eq)]
15pub struct VersionChange {
16    pub name: String,
17    pub old_version: String,
18    pub new_version: String,
19}
20
21/// Enter pre-release mode for the selected packages with the provided label.
22pub 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
40/// Exit pre-release mode for the selected packages, restoring stable versions.
41pub 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
54/// Restore any preserved changesets from a prior pre-release phase back into the
55/// workspace changeset directory.
56///
57/// Returns the number of files moved. When no preserved changesets are present,
58/// the function behaves as a no-op.
59pub 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}