diff --git a/src/cargo/util/toml_mut/dependency.rs b/src/cargo/util/toml_mut/dependency.rs index 78cc58a29b71..6d693d6dda39 100644 --- a/src/cargo/util/toml_mut/dependency.rs +++ b/src/cargo/util/toml_mut/dependency.rs @@ -1252,6 +1252,8 @@ mod tests { let mut local = LocalManifest { path: crate_root.clone(), manifest, + embedded: None, + raw: toml.to_owned(), }; assert_eq!(local.manifest.to_string(), toml); let gctx = GlobalContext::default().unwrap(); diff --git a/src/cargo/util/toml_mut/manifest.rs b/src/cargo/util/toml_mut/manifest.rs index 98033531e3ad..79fd24029ded 100644 --- a/src/cargo/util/toml_mut/manifest.rs +++ b/src/cargo/util/toml_mut/manifest.rs @@ -11,6 +11,7 @@ use crate::core::dependency::DepKind; use crate::core::{FeatureValue, Features, Workspace}; use crate::util::closest; use crate::util::interning::InternedString; +use crate::util::toml::{is_embedded, ScriptSource}; use crate::{CargoResult, GlobalContext}; /// Dependency table to add deps to. @@ -245,6 +246,10 @@ pub struct LocalManifest { pub path: PathBuf, /// Manifest contents. pub manifest: Manifest, + /// The raw, unparsed package file + pub raw: String, + /// Edit location for an embedded manifest, if relevant + pub embedded: Option, } impl Deref for LocalManifest { @@ -267,18 +272,56 @@ impl LocalManifest { if !path.is_absolute() { anyhow::bail!("can only edit absolute paths, got {}", path.display()); } - let data = cargo_util::paths::read(&path)?; + let raw = cargo_util::paths::read(&path)?; + let mut data = raw.clone(); + let mut embedded = None; + if is_embedded(path) { + let source = ScriptSource::parse(&data)?; + if let Some(frontmatter) = source.frontmatter() { + embedded = Some(Embedded::exists(&data, frontmatter)); + data = frontmatter.to_owned(); + } else if let Some(shebang) = source.shebang() { + embedded = Some(Embedded::after(&data, shebang)); + data = String::new(); + } else { + embedded = Some(Embedded::start()); + data = String::new(); + } + } let manifest = data.parse().context("Unable to parse Cargo.toml")?; Ok(LocalManifest { manifest, path: path.to_owned(), + raw, + embedded, }) } /// Write changes back to the file. pub fn write(&self) -> CargoResult<()> { - let s = self.manifest.data.to_string(); - let new_contents_bytes = s.as_bytes(); + let mut manifest = self.manifest.data.to_string(); + let raw = match self.embedded.as_ref() { + Some(Embedded::Implicit(start)) => { + if !manifest.ends_with("\n") { + manifest.push_str("\n"); + } + let fence = "---\n"; + let prefix = &self.raw[0..*start]; + let suffix = &self.raw[*start..]; + let empty_line = if prefix.is_empty() { "\n" } else { "" }; + format!("{prefix}{fence}{manifest}{fence}{empty_line}{suffix}") + } + Some(Embedded::Explicit(span)) => { + if !manifest.ends_with("\n") { + manifest.push_str("\n"); + } + let prefix = &self.raw[0..span.start]; + let suffix = &self.raw[span.end..]; + format!("{prefix}{manifest}{suffix}") + } + None => manifest, + }; + let new_contents_bytes = raw.as_bytes(); cargo_util::paths::write_atomic(&self.path, new_contents_bytes) } @@ -531,6 +574,51 @@ impl std::fmt::Display for LocalManifest { } } +/// Edit location for an embedded manifest +#[derive(Clone, Debug)] +pub enum Embedded { + /// Manifest is implicit + /// + /// This is the insert location for a frontmatter + Implicit(usize), + /// Manifest is explicit in a frontmatter + /// + /// This is the span of the frontmatter body + Explicit(std::ops::Range), +} + +impl Embedded { + fn start() -> Self { + Self::Implicit(0) + } + + fn after(input: &str, after: &str) -> Self { + let span = substr_span(input, after); + let end = span.end; + Self::Implicit(end) + } + + fn exists(input: &str, exists: &str) -> Self { + let span = substr_span(input, exists); + Self::Explicit(span) + } +} + +fn substr_span(haystack: &str, needle: &str) -> std::ops::Range { + let haystack_start_ptr = haystack.as_ptr(); + let haystack_end_ptr = haystack[haystack.len()..haystack.len()].as_ptr(); + + let needle_start_ptr = needle.as_ptr(); + let needle_end_ptr = needle[needle.len()..needle.len()].as_ptr(); + + assert!(needle_end_ptr < haystack_end_ptr); + assert!(haystack_start_ptr <= needle_start_ptr); + let start = needle_start_ptr as usize - haystack_start_ptr as usize; + let end = start + needle.len(); + + start..end +} + #[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord)] enum DependencyStatus { None, diff --git a/tests/testsuite/cargo_add/script_bare/mod.rs b/tests/testsuite/cargo_add/script_bare/mod.rs index 3b0f34317ab6..ffa6db6c6f87 100644 --- a/tests/testsuite/cargo_add/script_bare/mod.rs +++ b/tests/testsuite/cargo_add/script_bare/mod.rs @@ -31,7 +31,7 @@ fn case() { .arg_line("--manifest-path cargo-test-fixture.rs my-package") .current_dir(cwd) .assert() - .failure() + .success() .stdout_eq(str![""]) .stderr_eq(file!["stderr.term.svg"]); diff --git a/tests/testsuite/cargo_add/script_bare/out/cargo-test-fixture.rs b/tests/testsuite/cargo_add/script_bare/out/cargo-test-fixture.rs index f328e4d9d04c..caf9ce19959e 100644 --- a/tests/testsuite/cargo_add/script_bare/out/cargo-test-fixture.rs +++ b/tests/testsuite/cargo_add/script_bare/out/cargo-test-fixture.rs @@ -1 +1,6 @@ +--- +[dependencies] +my-package = "99999.0.0" +--- + fn main() {} diff --git a/tests/testsuite/cargo_add/script_bare/stderr.term.svg b/tests/testsuite/cargo_add/script_bare/stderr.term.svg index f0187d20a34f..b5c72375b88c 100644 --- a/tests/testsuite/cargo_add/script_bare/stderr.term.svg +++ b/tests/testsuite/cargo_add/script_bare/stderr.term.svg @@ -1,8 +1,8 @@ - +