Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merge tools definition recursively #1353

Merged
merged 2 commits into from
Jun 17, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions scarb/src/core/manifest/toml_manifest.rs
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ use crate::core::{
};
use crate::internal::fsx;
use crate::internal::fsx::PathBufUtf8Ext;
use crate::internal::serdex::{toml_merge, RelativeUtf8PathBuf};
use crate::internal::serdex::{toml_merge, toml_merge_apply_strategy, RelativeUtf8PathBuf};
use crate::internal::to_version::ToVersion;
use crate::{
DEFAULT_MODULE_MAIN_FILE, DEFAULT_SOURCE_PATH, DEFAULT_TESTS_PATH, MANIFEST_FILE_NAME,
Expand Down Expand Up @@ -876,7 +876,7 @@ impl TomlManifest {
.transpose()?
.map(|tool| {
if let Some(profile_tool) = &profile_definition.tool {
toml_merge(&tool, profile_tool)
toml_merge_apply_strategy(&tool, profile_tool)
} else {
Ok(tool)
}
Expand Down
158 changes: 155 additions & 3 deletions scarb/src/internal/serdex.rs
Original file line number Diff line number Diff line change
@@ -1,11 +1,93 @@
use anyhow::Result;
use anyhow::{anyhow, Result};
use camino::{Utf8Path, Utf8PathBuf};
use serde::{Deserialize, Serialize};

use crate::internal::fsx;

/// Merge two `toml::Value` serializable structs.
pub fn toml_merge<'de, T, S>(target: &T, source: &S) -> anyhow::Result<T>
const MERGE_STRATEGY_KEY: &str = "merge-strategy";

#[derive(Debug, Default)]
pub enum MergeStrategy {
#[default]
Override,
Merge,
}

impl TryFrom<&str> for MergeStrategy {
type Error = anyhow::Error;
fn try_from(value: &str) -> Result<Self, Self::Error> {
match value {
"override" => Ok(Self::Override),
"merge" => Ok(Self::Merge),
_ => Err(anyhow!(
"invalid merge strategy: {}, must be one of `merge`, `override`",
value
)),
}
}
}

/// Merge `source` into `target` where `source` and `target` are two `toml::Value` serializable
/// structs.
/// If `source` and `target` are tables and a specific key exists in both, the conflict will be
/// resolved as follows:
/// 1) If value under the conflicting key in either `source` or `target` is not a table, the value
/// in `target` will be overridden with the value from `source`. This means that no keys from
/// subtables under the conflicting key will be preserved.
/// 2) If values under the conflicting key in both `source` and `target` are tables,
/// the conflict will be resolved with one of two strategies defined by `source`:
/// a) If `source` have a key `merge-strategy` with value `override`, the value in `target` will
/// be overridden with the value from `source`.
/// b) If `source` have a key `merge-strategy` with value `merge`, the value in `target` will be
/// merged with the value from `source` recursively with this function.
/// c) If `source` does not have a key `merge-strategy`, the value in `target` will be overridden.
maciektr marked this conversation as resolved.
Show resolved Hide resolved
pub fn toml_merge_apply_strategy<'de, T, S>(target: &T, source: &S) -> Result<T>
where
maciektr marked this conversation as resolved.
Show resolved Hide resolved
maciektr marked this conversation as resolved.
Show resolved Hide resolved
T: Serialize + Deserialize<'de>,
S: Serialize + Deserialize<'de>,
{
let mut target_value = toml::Value::try_from(target)?;
let source_value = toml::Value::try_from(source)?;

if let (Some(target_table), Some(source_table)) =
(target_value.as_table_mut(), source_value.as_table())
{
for (key, value) in source_table {
match target_table.get_mut(key) {
Some(target_subvalue) if target_subvalue.is_table() && value.is_table() => {
let target_subtable = target_subvalue.as_table_mut().unwrap();
let value_subtable = value.as_table().unwrap();
let strategy = value_subtable
.get(MERGE_STRATEGY_KEY)
.and_then(|v| v.as_str())
.map_or(Ok(MergeStrategy::default()), MergeStrategy::try_from)?;
match &strategy {
MergeStrategy::Override => {
*target_subvalue = toml::Value::try_from(value_subtable.clone())?;
}
MergeStrategy::Merge => {
*target_subvalue = toml::Value::try_from(toml_merge_apply_strategy(
target_subtable,
value_subtable,
)?)?;
}
}
}
_ => {
target_table.insert(key.clone(), value.clone());
}
}
}
}

Ok(toml::Value::try_into(target_value)?)
}

/// Merge `source` into `target` where `source` and `target` are two `toml::Value` serializable
/// structs.
/// If `source` and `target` are tables and a specific key exists in both, the value in `target`
/// will be overridden with the value from `source`.
pub fn toml_merge<'de, T, S>(target: &T, source: &S) -> Result<T>
where
T: Serialize + Deserialize<'de>,
S: Serialize + Deserialize<'de>,
Expand Down Expand Up @@ -39,3 +121,73 @@ impl RelativeUtf8PathBuf {
self.relative_to_directory(root)
}
}

#[cfg(test)]
mod tests {
use super::{toml_merge, toml_merge_apply_strategy};
use test_case::test_case;

#[test_case(r#"{}"#, r#"{}"#)]
#[test_case(r#"{"a": "a"}"#, r#"{"b":"b"}"#)]
#[test_case(r#"{"a": "a"}"#, r#"{"a":"b"}"#)]
#[test_case(r#"{"a": {"a": "a"}}"#, r#"{"a": {"a": "b"}}"#)]
#[test_case(
r#"{"a": {"a": "a", "merge-strategy": "merge"}}"#,
r#"{"a": {"a": "b"}}"#
)]
#[test_case(
r#"{"a": {"a": "a", "merge-strategy": "override"}}"#,
r#"{"a": {"a": "b", "b": "b"}}"#
)]
fn merge_with_override(source: &'static str, target: &'static str) {
let source: toml::Value = serde_json::from_str(source).unwrap();
let target: toml::Value = serde_json::from_str(target).unwrap();
assert_eq!(
toml_merge(&target, &source).unwrap(),
toml_merge_apply_strategy(&target, &source).unwrap()
);
}

#[test]
fn merge_with_merge_strategy() {
let target = r#"{"a": {"a": "b", "b": "b"}}"#;
let source = r#"{"a": {"a": "a", "merge-strategy": "merge"}}"#;
let target: toml::Value = serde_json::from_str(target).unwrap();
let source: toml::Value = serde_json::from_str(source).unwrap();
let with_override = toml_merge(&target, &source).unwrap();
let with_merge = toml_merge_apply_strategy(&target, &source).unwrap();
let with_override = serde_json::to_string(&with_override).unwrap();
let with_merge = serde_json::to_string(&with_merge).unwrap();
assert_eq!(with_override, r#"{"a":{"a":"a","merge-strategy":"merge"}}"#);
assert_eq!(
with_merge,
r#"{"a":{"a":"a","b":"b","merge-strategy":"merge"}}"#
);
}

#[test_case(
r#"{"a": {"a": "b", "b": "b"}}"#,
r#"{"a": {"a": "a"}, "merge-strategy": "merge"}"#
)]
#[test_case(
r#"{"a": {"merge-strategy": "merge", "a": "b", "b": "b"}}"#,
r#"{"a": {"a": "a"}}"#
)]
fn merge_strategy_must_be_on_source(target: &'static str, source: &'static str) {
let source: toml::Value = serde_json::from_str(source).unwrap();
let target: toml::Value = serde_json::from_str(target).unwrap();
assert_eq!(
toml_merge(&target, &source).unwrap(),
toml_merge_apply_strategy(&target, &source).unwrap()
);
}

#[test]
fn invalid_merge_strategy() {
let target = r#"{"a": {"a": "b", "b": "b"}}"#;
let source = r#"{"a": {"a": "a", "merge-strategy": "other"}}"#;
let target: toml::Value = serde_json::from_str(target).unwrap();
let source: toml::Value = serde_json::from_str(source).unwrap();
assert!(toml_merge_apply_strategy(&target, &source).is_err());
}
}
68 changes: 66 additions & 2 deletions scarb/tests/profiles.rs
Original file line number Diff line number Diff line change
Expand Up @@ -610,6 +610,9 @@ fn profile_overrides_tool() {

[profile.release.tool.snforge]
some-key = "some-other-value"

[profile.newprof]
inherits = "release"
"#})
.dep_cairo_test()
.build(&t);
Expand Down Expand Up @@ -641,11 +644,17 @@ fn profile_overrides_tool() {
);

let metadata = Scarb::quick_snapbox()
.args(["--json", "--release", "metadata", "--format-version", "1"])
.args([
"--json",
"--profile=newprof",
"metadata",
"--format-version",
"1",
])
.current_dir(&t)
.stdout_json::<Metadata>();

assert_eq!(metadata.current_profile, "release".to_string());
assert_eq!(metadata.current_profile, "newprof".to_string());
assert_eq!(metadata.packages.len(), 3);
let package = metadata
.packages
Expand All @@ -667,3 +676,58 @@ fn profile_overrides_tool() {
"some-other-value"
);
}

#[test]
fn tools_can_be_merged_recursively() {
let t = TempDir::new().unwrap();
ProjectBuilder::start()
.name("hello")
.manifest_extra(indoc! {r#"
[tool.something.a]
key = "value"

[profile.release]
inherits = "dev"

[profile.release.tool.something]
merge-strategy = "merge"
b = "some-value"
"#})
.build(&t);

let metadata = Scarb::quick_snapbox()
.args([
"--json",
"--profile=release",
"metadata",
"--format-version",
"1",
])
.current_dir(&t)
.stdout_json::<Metadata>();
let package = metadata
.packages
.iter()
.find(|pkg| pkg.name.starts_with("hello"))
.cloned()
.unwrap();
assert_eq!(
package
.manifest_metadata
.tool
.unwrap()
.get("something")
.unwrap()
.as_object()
.unwrap()
.get("a")
.unwrap()
.as_object()
.unwrap()
.get("key")
.unwrap()
.as_str()
.unwrap(),
"value"
);
}
Loading
Loading