1use crate::config;
2use crate::error::CliError;
3use crate::ops::cargo;
4use crate::ops::git;
5use crate::steps::plan;
6
7#[derive(Debug, Clone, clap::Args)]
8pub struct ReleaseStep {
9 #[command(flatten)]
10 manifest: clap_cargo::Manifest,
11
12 #[command(flatten)]
13 workspace: clap_cargo::Workspace,
14
15 #[arg(long, conflicts_with = "level_or_version")]
17 unpublished: bool,
18
19 #[arg(value_name = "LEVEL|VERSION")]
21 level_or_version: Option<super::TargetVersion>,
22
23 #[arg(short, long, requires = "level_or_version")]
25 metadata: Option<String>,
26
27 #[arg(short = 'x', long)]
29 execute: bool,
30
31 #[arg(short = 'n', long, conflicts_with = "execute", hide = true)]
32 dry_run: bool,
33
34 #[arg(long)]
36 no_confirm: bool,
37
38 #[arg(long, value_name = "NAME")]
40 prev_tag_name: Option<String>,
41
42 #[command(flatten)]
43 config: config::ConfigArgs,
44}
45
46impl ReleaseStep {
47 pub fn run(&self) -> Result<(), CliError> {
48 git::git_version()?;
49 let mut index = crate::ops::index::CratesIoIndex::new();
50
51 if self.dry_run {
52 let _ =
53 crate::ops::shell::warn("`--dry-run` is superfluous, dry-run is done by default");
54 }
55
56 let ws_meta = self
57 .manifest
58 .metadata()
59 .features(cargo_metadata::CargoOpt::AllFeatures)
61 .exec()?;
62 let mut ws_config = config::load_workspace_config(&self.config, &ws_meta)?;
63 let mut pkgs = plan::load(&self.config, &ws_meta)?;
64
65 for pkg in pkgs.values_mut() {
66 if let Some(prev_tag) = self.prev_tag_name.as_ref() {
67 pkg.set_prior_tag(prev_tag.to_owned());
70 }
71 if pkg.config.release()
72 && let Some(level_or_version) = &self.level_or_version
73 {
74 pkg.bump(level_or_version, self.metadata.as_deref())?;
75 }
76 if index.has_krate(
77 pkg.config.registry(),
78 &pkg.meta.name,
79 pkg.config.certs_source(),
80 )? {
81 pkg.ensure_owners = false;
83 }
84 }
85
86 let (_selected_pkgs, excluded_pkgs) =
87 if self.unpublished && self.workspace == clap_cargo::Workspace::default() {
88 ws_meta.packages.iter().partition(|_| false)
89 } else {
90 self.workspace.partition_packages(&ws_meta)
91 };
92 for excluded_pkg in &excluded_pkgs {
93 let Some(pkg) = pkgs.get_mut(&excluded_pkg.id) else {
94 continue;
96 };
97 if !pkg.config.release() {
98 continue;
99 }
100
101 let crate_name = pkg.meta.name.as_str();
102 let explicitly_excluded = self.workspace.exclude.contains(&excluded_pkg.name);
103 if pkg.config.release()
106 && pkg.config.publish()
107 && self.unpublished
108 && !explicitly_excluded
109 {
110 let version = &pkg.initial_version;
111 if !cargo::is_published(
112 &mut index,
113 pkg.config.registry(),
114 crate_name,
115 &version.full_version_string,
116 pkg.config.certs_source(),
117 ) {
118 log::debug!(
119 "enabled {}, v{} is unpublished",
120 crate_name,
121 version.full_version_string
122 );
123 continue;
124 }
125 }
126
127 pkg.planned_version = None;
128 pkg.config.release = Some(false);
129
130 if let Some(prior_tag_name) = &pkg.prior_tag {
131 if let Some(changed) =
132 crate::steps::version::changed_since(&ws_meta, pkg, prior_tag_name)
133 {
134 if !changed.is_empty() {
135 let _ = crate::ops::shell::warn(format!(
136 "disabled by user, skipping {crate_name} which has files changed since {prior_tag_name}: {changed:#?}"
137 ));
138 } else {
139 log::trace!(
140 "disabled by user, skipping {crate_name} (no changes since {prior_tag_name})"
141 );
142 }
143 } else {
144 log::debug!(
145 "disabled by user, skipping {crate_name} (no {prior_tag_name} tag)"
146 );
147 }
148 } else {
149 log::debug!("disabled by user, skipping {crate_name} (no tag found)",);
150 }
151 }
152
153 let pkgs = plan::plan(pkgs)?;
154
155 for excluded_pkg in &excluded_pkgs {
156 let Some(pkg) = pkgs.get(&excluded_pkg.id) else {
157 continue;
159 };
160
161 if pkg.config.publish() && pkg.config.registry().is_none() {
163 let version = pkg.planned_version.as_ref().unwrap_or(&pkg.initial_version);
164 let crate_name = pkg.meta.name.as_str();
165 if !cargo::is_published(
166 &mut index,
167 pkg.config.registry(),
168 crate_name,
169 &version.full_version_string,
170 pkg.config.certs_source(),
171 ) {
172 let _ = crate::ops::shell::warn(format!(
173 "disabled by user, skipping {} v{} despite being unpublished",
174 crate_name, version.full_version_string,
175 ));
176 }
177 }
178 }
179
180 let (selected_pkgs, excluded_pkgs): (Vec<_>, Vec<_>) = pkgs
181 .into_iter()
182 .map(|(_, pkg)| pkg)
183 .partition(|p| p.config.release());
184 if selected_pkgs.is_empty() {
185 let _ = crate::ops::shell::error("no packages selected");
186 return Err(2.into());
187 }
188
189 let dry_run = !self.execute;
190 let mut failed = false;
191
192 let consolidate_commits = super::consolidate_commits(&selected_pkgs, &excluded_pkgs)?;
193 ws_config.consolidate_commits = Some(consolidate_commits);
194
195 failed |= !super::verify_git_is_clean(
197 ws_meta.workspace_root.as_std_path(),
198 dry_run,
199 log::Level::Error,
200 )?;
201
202 failed |= !super::verify_tags_missing(&selected_pkgs, dry_run, log::Level::Error)?;
203
204 failed |=
205 !super::verify_monotonically_increasing(&selected_pkgs, dry_run, log::Level::Error)?;
206
207 let mut double_publish = false;
208 for pkg in &selected_pkgs {
209 if !pkg.config.publish() {
210 continue;
211 }
212 let version = pkg.planned_version.as_ref().unwrap_or(&pkg.initial_version);
213 let crate_name = pkg.meta.name.as_str();
214 if cargo::is_published(
215 &mut index,
216 pkg.config.registry(),
217 crate_name,
218 &version.full_version_string,
219 pkg.config.certs_source(),
220 ) {
221 let registry = pkg.config.registry().unwrap_or("crates.io");
222 let _ = crate::ops::shell::error(format!(
223 "{} {} is already published to {}",
224 crate_name, version.full_version_string, registry
225 ));
226 double_publish = true;
227 }
228 }
229 if double_publish {
230 failed = true;
231 if !dry_run {
232 return Err(101.into());
233 }
234 }
235
236 super::warn_changed(&ws_meta, &selected_pkgs)?;
237
238 failed |= !super::verify_git_branch(
239 ws_meta.workspace_root.as_std_path(),
240 &ws_config,
241 dry_run,
242 log::Level::Error,
243 )?;
244
245 failed |= !super::verify_if_behind(
246 ws_meta.workspace_root.as_std_path(),
247 &ws_config,
248 dry_run,
249 log::Level::Warn,
250 )?;
251
252 failed |= !super::verify_metadata(&selected_pkgs, dry_run, log::Level::Error)?;
253 failed |= !super::verify_rate_limit(
254 &selected_pkgs,
255 &mut index,
256 &ws_config.rate_limit,
257 dry_run,
258 log::Level::Error,
259 )?;
260
261 super::confirm("Release", &selected_pkgs, self.no_confirm, dry_run)?;
263
264 if consolidate_commits {
266 let update_lock =
267 super::version::update_versions(&ws_meta, &selected_pkgs, &excluded_pkgs, dry_run)?;
268 if update_lock {
269 log::debug!("updating lock file");
270 if !dry_run {
271 let workspace_path = ws_meta.workspace_root.as_std_path().join("Cargo.toml");
272 cargo::update_lock(&workspace_path)?;
273 }
274 }
275
276 for pkg in &selected_pkgs {
277 super::replace::replace(pkg, dry_run)?;
278
279 super::hook::hook(&ws_meta, pkg, dry_run)?;
281 }
282
283 super::commit::workspace_commit(&ws_meta, &ws_config, &selected_pkgs, dry_run)?;
284 } else {
285 for pkg in &selected_pkgs {
286 if let Some(version) = pkg.planned_version.as_ref() {
287 let crate_name = pkg.meta.name.as_str();
288 let _ = crate::ops::shell::status(
289 "Upgrading",
290 format!(
291 "{} from {} to {}",
292 crate_name,
293 pkg.initial_version.full_version_string,
294 version.full_version_string
295 ),
296 );
297 cargo::set_package_version(
298 &pkg.manifest_path,
299 version.full_version_string.as_str(),
300 dry_run,
301 )?;
302 crate::steps::version::update_dependent_versions(
303 &ws_meta, pkg, version, dry_run,
304 )?;
305 if dry_run {
306 log::debug!("updating lock file");
307 } else {
308 cargo::update_lock(&pkg.manifest_path)?;
309 }
310 }
311
312 super::replace::replace(pkg, dry_run)?;
313
314 super::hook::hook(&ws_meta, pkg, dry_run)?;
316
317 super::commit::pkg_commit(pkg, dry_run)?;
318 }
319 }
320
321 super::publish::publish(&selected_pkgs, dry_run, &ws_config.unstable)?;
323 super::owner::ensure_owners(&selected_pkgs, dry_run)?;
324
325 super::tag::tag(&selected_pkgs, dry_run)?;
327
328 super::push::push(&ws_config, &ws_meta, &selected_pkgs, dry_run)?;
330
331 super::finish(failed, dry_run)
332 }
333}