|
| 1 | +//! Functionality related to deleting a crate. |
| 2 | +
|
| 3 | +use std::sync::Arc; |
| 4 | + |
| 5 | +use chrono::{Duration, Utc}; |
| 6 | +use diesel::dsl::count_star; |
| 7 | + |
| 8 | +use crate::controllers::cargo_prelude::*; |
| 9 | + |
| 10 | +use crate::schema::*; |
| 11 | +use crate::util::errors::internal; |
| 12 | +use crate::views::{EncodableCrate, GoodCrate, PublishWarnings}; |
| 13 | + |
| 14 | +use super::{extract_crate_name_and_semver, version_and_crate}; |
| 15 | + |
| 16 | +/// Handles the `DELETE /crates/:crate_id/:version` route. |
| 17 | +/// |
| 18 | +/// Actually deletion is allowed only in the first 24 hours from creation |
| 19 | +pub fn delete(req: &mut dyn RequestExt) -> EndpointResult { |
| 20 | + let app = Arc::clone(req.app()); |
| 21 | + let (crate_name, semver) = extract_crate_name_and_semver(req)?; |
| 22 | + let conn = req.db_read()?; |
| 23 | + let (version, krate) = version_and_crate(&conn, crate_name, semver)?; |
| 24 | + |
| 25 | + if Utc::now() |
| 26 | + .naive_utc() |
| 27 | + .signed_duration_since(version.created_at) |
| 28 | + > Duration::hours(24) |
| 29 | + { |
| 30 | + return Err(cargo_err( |
| 31 | + "Version deletion is allowed only in the first 24 hours from creation", |
| 32 | + )); |
| 33 | + } |
| 34 | + |
| 35 | + // Create a transaction on the database, if there are no errors, |
| 36 | + // commit the transactions to delete the version. |
| 37 | + // If there are no remaining versions, delete the crate |
| 38 | + conn.transaction(|| { |
| 39 | + diesel::delete(versions::table.find(version.id)).execute(&*conn)?; |
| 40 | + |
| 41 | + let top_versions = krate.top_versions(&conn)?; |
| 42 | + |
| 43 | + // we can't check top_versions to know if there aren't remaining versions |
| 44 | + // because it excludes yanked versions |
| 45 | + let remaining: i64 = versions::table |
| 46 | + .filter(versions::crate_id.eq(krate.id)) |
| 47 | + .select(count_star()) |
| 48 | + .first(&*conn) |
| 49 | + .optional()? |
| 50 | + .unwrap_or_default(); |
| 51 | + if remaining <= 0 { |
| 52 | + diesel::delete(crates::table.find(krate.id)).execute(&*conn)?; |
| 53 | + |
| 54 | + let uploader = app.config.uploader(); |
| 55 | + uploader |
| 56 | + .delete_index(app.http_client(), &krate.name) |
| 57 | + .map_err(|e| internal(&format_args!("failed to delete crate: {}", e)))?; |
| 58 | + } |
| 59 | + |
| 60 | + Ok(req.json(&GoodCrate { |
| 61 | + krate: EncodableCrate::from_minimal(krate, Some(&top_versions), None, true, None), |
| 62 | + warnings: PublishWarnings { |
| 63 | + invalid_categories: vec![], |
| 64 | + invalid_badges: vec![], |
| 65 | + other: vec![], |
| 66 | + }, |
| 67 | + })) |
| 68 | + }) |
| 69 | +} |
0 commit comments