aboutsummaryrefslogtreecommitdiffstats
path: root/xtask/src/release.rs
blob: 3e8f02e893d738976e59c78a8db552b84b3cebb3 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
use std::process::{Command, Stdio};

use anyhow::Result;
use clap::{Args, Subcommand};
use semver::Version;

use crate::PKG_VER;

use self::bump::{Bump, Level};

mod bump;

#[derive(Debug, Clone, Subcommand)]
pub enum Step {
    /// Bump version in package files and commit changes.
    Bump {
        /// Level of version bump version.
        #[arg(required = false)]
        level: bump::Level,
    },

    /// Make a release commit.
    Commit {
        /// Options passed to git commit.
        #[arg(last = true)]
        git_commit_args: Vec<String>,
    },

    /// Create git tag for release.
    Tag,
}

#[derive(Debug, Clone, Args)]
pub struct Release {
    #[command(subcommand)]
    step: Step,
}

impl Release {
    pub fn run(self) -> Result<()> {
        match self.step {
            Step::Bump { level } => {
                let bump = Self::bump(level)?;
                println!("Bumped version: {bump}");
            }
            Step::Commit { git_commit_args } => {
                let version = PKG_VER.parse()?;
                Self::commit(version, git_commit_args)?
            }
            Step::Tag => {
                let stdout = Command::new("git")
                    .arg("describe")
                    .arg("--abbrev=0")
                    .output()?
                    .stdout;

                let prev = std::str::from_utf8(&stdout)?
                    .trim()
                    .trim_start_matches('v')
                    .parse()?;

                let next = PKG_VER.parse()?;
                Self::tag(prev, next)?;
            }
        };

        Ok(())
    }

    pub fn bump(level: Level) -> Result<Bump> {
        let version = PKG_VER.parse().unwrap_or_else(|_| Version::new(0, 1, 0));
        let next = level.bump(&version);

        let mut bump = Bump { next, version };

        bump.bump_file("./Cargo.toml", bump::cargo)?;
        bump.bump_file("./README.md", bump::replace)?;
        bump.bump_file("./CHANGELOG.md", bump::changelog)?;
        bump.bump_file("./pkg/archlinux/projectr/PKGBUILD", bump::replace)?;
        bump.bump_file("./pkg/archlinux/projectr-bin/PKGBUILD", bump::replace)?;
        bump.bump_file("./pkg/archlinux/projectr-git/PKGBUILD", bump::vsc_pkgbuild)?;

        let cargo_update = Command::new("cargo")
            .arg("update")
            .arg("--workspace")
            .status()?;
        anyhow::ensure!(cargo_update.success(), "Failed to update cargo lockfile");

        let git_added = Command::new("git")
            .arg("add")
            .arg("./Cargo.lock")
            .status()?;
        anyhow::ensure!(git_added.success(), "Failed to add Cargo.lock to git");

        Ok(bump)
    }

    pub fn commit(version: Version, git_commit_args: Vec<String>) -> Result<()> {
        let git_commit = Command::new("git")
            .arg("commit")
            .arg("-em")
            .arg(format!("chore: release projectr version {version}"))
            .args(git_commit_args)
            .status()?;

        anyhow::ensure!(git_commit.success(), "Failed to commit changes");

        Ok(())
    }

    pub fn tag(from: Version, to: Version) -> Result<String> {
        let tag_name = format!("v{}", to);

        let shortlog_child = Command::new("git")
            .arg("shortlog")
            .arg(format!("v{}..HEAD", from))
            .arg("--abbrev=7")
            .stdout(Stdio::piped())
            .spawn()?;

        let git_commit = Command::new("git")
            .arg("tag")
            .arg("-s")
            .arg(&tag_name)
            .arg("--file")
            .arg("-")
            .stdin(Stdio::from(shortlog_child.stdout.unwrap())) // Pipe through.
            .status()?;

        anyhow::ensure!(git_commit.success(), "Failed to commit changes");

        Ok(tag_name)
    }
}