sampo_core/
prerelease.rs

1use crate::adapters::{ManifestMetadata, PackageAdapter};
2use crate::discover_workspace;
3use crate::errors::{Result, SampoError};
4use crate::release::{parse_version_string, regenerate_lockfile, restore_prerelease_changesets};
5use crate::types::{
6    PackageInfo, PackageKind, 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 has_cargo = workspace
237        .members
238        .iter()
239        .any(|pkg| pkg.kind == PackageKind::Cargo);
240    let manifest_metadata = if has_cargo {
241        Some(ManifestMetadata::load(workspace).map_err(|err| match err {
242            SampoError::Release(msg) => SampoError::Prerelease(msg),
243            other => other,
244        })?)
245    } else {
246        None
247    };
248
249    for info in &workspace.members {
250        let adapter = match info.kind {
251            PackageKind::Cargo => PackageAdapter::Cargo,
252            PackageKind::Npm => PackageAdapter::Npm,
253        };
254        let manifest_path = adapter.manifest_path(&info.path);
255        let original = fs::read_to_string(&manifest_path)?;
256        let new_pkg_version = new_versions.get(&info.name).map(|s| s.as_str());
257        let metadata_ref = match info.kind {
258            PackageKind::Cargo => manifest_metadata.as_ref(),
259            PackageKind::Npm => None,
260        };
261        let (updated, _deps) = adapter.update_manifest_versions(
262            &manifest_path,
263            &original,
264            new_pkg_version,
265            new_versions,
266            metadata_ref,
267        )?;
268
269        if updated != original {
270            fs::write(&manifest_path, updated)?;
271        }
272    }
273
274    regenerate_lockfile(workspace).map_err(|err| match err {
275        SampoError::Release(msg) => SampoError::Prerelease(msg),
276        other => other,
277    })?;
278
279    Ok(())
280}
281
282#[cfg(test)]
283mod tests {
284    use super::*;
285    use tempfile::tempdir;
286
287    fn init_workspace() -> tempfile::TempDir {
288        let temp = tempdir().unwrap();
289        let root = temp.path();
290
291        fs::create_dir_all(root.join("crates/foo")).unwrap();
292        fs::create_dir_all(root.join("crates/bar")).unwrap();
293
294        fs::write(
295            root.join("Cargo.toml"),
296            "[workspace]\nmembers=[\"crates/*\"]\n",
297        )
298        .unwrap();
299
300        write_manifest(&root.join("crates/foo"), "foo", "0.1.0");
301        write_manifest(&root.join("crates/bar"), "bar", "0.1.0");
302
303        temp
304    }
305
306    fn write_manifest(path: &Path, name: &str, version: &str) {
307        fs::create_dir_all(path.join("src")).unwrap();
308        fs::write(
309            path.join("Cargo.toml"),
310            format!("[package]\nname = \"{name}\"\nversion = \"{version}\"\n"),
311        )
312        .unwrap();
313        fs::write(path.join("src/lib.rs"), "pub fn __sampo_test_marker() {}\n").unwrap();
314    }
315
316    fn append_dependency(path: &Path, dep: &str, dep_version: &str) {
317        let manifest_path = path.join("Cargo.toml");
318        let current = fs::read_to_string(&manifest_path).unwrap();
319        fs::write(
320            &manifest_path,
321            format!(
322                "{}\n[dependencies]\n{dep} = {{ path = \"../{dep}\", version = \"{dep_version}\" }}\n",
323                current.trim_end()
324            ),
325        )
326        .unwrap();
327    }
328
329    #[test]
330    fn enter_sets_prerelease_label_and_updates_dependents() {
331        let temp = init_workspace();
332        let root = temp.path();
333
334        write_manifest(&root.join("crates/foo"), "foo", "1.2.3");
335        write_manifest(&root.join("crates/bar"), "bar", "0.1.0");
336        append_dependency(&root.join("crates/bar"), "foo", "1.2.3");
337
338        let updates = enter_prerelease(root, &[String::from("foo")], "alpha").unwrap();
339        assert_eq!(
340            updates,
341            vec![VersionChange {
342                name: "foo".to_string(),
343                old_version: "1.2.3".to_string(),
344                new_version: "1.2.3-alpha".to_string(),
345            }]
346        );
347
348        let foo_manifest = fs::read_to_string(root.join("crates/foo/Cargo.toml")).unwrap();
349        assert!(
350            foo_manifest.contains("version = \"1.2.3-alpha\"")
351                || foo_manifest.contains("version=\"1.2.3-alpha\"")
352        );
353
354        let bar_manifest = fs::read_to_string(root.join("crates/bar/Cargo.toml")).unwrap();
355        assert!(
356            bar_manifest.contains("version = \"1.2.3-alpha\"")
357                || bar_manifest.contains("version=\"1.2.3-alpha\"")
358        );
359    }
360
361    #[test]
362    fn enter_switches_between_labels() {
363        let temp = init_workspace();
364        let root = temp.path();
365
366        write_manifest(&root.join("crates/foo"), "foo", "1.0.0-beta.3");
367
368        let updates = enter_prerelease(root, &[String::from("foo")], "alpha").unwrap();
369        assert_eq!(
370            updates,
371            vec![VersionChange {
372                name: "foo".to_string(),
373                old_version: "1.0.0-beta.3".to_string(),
374                new_version: "1.0.0-alpha".to_string(),
375            }]
376        );
377
378        let foo_manifest = fs::read_to_string(root.join("crates/foo/Cargo.toml")).unwrap();
379        assert!(foo_manifest.contains("1.0.0-alpha"));
380    }
381
382    #[test]
383    fn enter_rejects_numeric_only_label() {
384        let temp = init_workspace();
385        let root = temp.path();
386
387        write_manifest(&root.join("crates/foo"), "foo", "0.1.0");
388
389        let err = enter_prerelease(root, &[String::from("foo")], "123").unwrap_err();
390        match err {
391            SampoError::Prerelease(msg) => {
392                assert!(msg.contains("non-numeric"));
393            }
394            other => panic!("unexpected error: {other:?}"),
395        }
396    }
397
398    #[test]
399    fn exit_clears_prerelease_and_updates_dependents() {
400        let temp = init_workspace();
401        let root = temp.path();
402
403        write_manifest(&root.join("crates/foo"), "foo", "2.3.4-alpha.5");
404        write_manifest(&root.join("crates/bar"), "bar", "0.2.0");
405        append_dependency(&root.join("crates/bar"), "foo", "2.3.4-alpha.5");
406
407        let updates = exit_prerelease(root, &[String::from("foo")]).unwrap();
408        assert_eq!(
409            updates,
410            vec![VersionChange {
411                name: "foo".to_string(),
412                old_version: "2.3.4-alpha.5".to_string(),
413                new_version: "2.3.4".to_string(),
414            }]
415        );
416
417        let foo_manifest = fs::read_to_string(root.join("crates/foo/Cargo.toml")).unwrap();
418        assert!(
419            foo_manifest.contains("version = \"2.3.4\"")
420                || foo_manifest.contains("version=\"2.3.4\"")
421        );
422
423        let bar_manifest = fs::read_to_string(root.join("crates/bar/Cargo.toml")).unwrap();
424        assert!(
425            bar_manifest.contains("version = \"2.3.4\"")
426                || bar_manifest.contains("version=\"2.3.4\"")
427        );
428    }
429
430    #[test]
431    fn restore_preserved_changesets_moves_files() {
432        let temp = init_workspace();
433        let root = temp.path();
434
435        let prerelease_dir = root.join(".sampo/prerelease");
436        fs::create_dir_all(&prerelease_dir).unwrap();
437        fs::write(prerelease_dir.join("change.md"), "---\nfoo: minor\n---\n").unwrap();
438
439        let restored = restore_preserved_changesets(root).unwrap();
440        assert_eq!(restored, 1);
441
442        let changesets_dir = root.join(".sampo/changesets");
443        let restored_entries = fs::read_dir(&changesets_dir)
444            .unwrap()
445            .map(|entry| entry.unwrap().path())
446            .collect::<Vec<_>>();
447        assert_eq!(restored_entries.len(), 1);
448        assert!(
449            restored_entries[0]
450                .file_name()
451                .unwrap()
452                .to_string_lossy()
453                .starts_with("change")
454        );
455
456        let remaining = fs::read_dir(&prerelease_dir).unwrap().collect::<Vec<_>>();
457        assert!(remaining.is_empty());
458    }
459}