Skip to content

Commit

Permalink
Merge pull request #606 from epage/ser
Browse files Browse the repository at this point in the history
fix(toml): Don't lose data when re-formatting
  • Loading branch information
epage authored Sep 9, 2023
2 parents 39eea9e + b2879d6 commit 025e40e
Show file tree
Hide file tree
Showing 2 changed files with 124 additions and 13 deletions.
32 changes: 19 additions & 13 deletions crates/toml/src/fmt.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
#[derive(Copy, Clone, Default)]
pub(crate) struct DocumentFormatter {
pub(crate) multiline_array: bool,
is_value: bool,
}

impl toml_edit::visit_mut::VisitMut for DocumentFormatter {
Expand All @@ -9,21 +10,26 @@ impl toml_edit::visit_mut::VisitMut for DocumentFormatter {
}

fn visit_item_mut(&mut self, node: &mut toml_edit::Item) {
let other = std::mem::take(node);
let other = match other.into_table().map(toml_edit::Item::Table) {
Ok(i) => i,
Err(i) => i,
};
let other = match other
.into_array_of_tables()
.map(toml_edit::Item::ArrayOfTables)
{
Ok(i) => i,
Err(i) => i,
};
*node = other;
let is_parent_value = self.is_value;
if !is_parent_value {
let other = std::mem::take(node);
let other = match other.into_table().map(toml_edit::Item::Table) {
Ok(i) => i,
Err(i) => i,
};
let other = match other
.into_array_of_tables()
.map(toml_edit::Item::ArrayOfTables)
{
Ok(i) => i,
Err(i) => i,
};
self.is_value = other.is_value();
*node = other;
}

toml_edit::visit_mut::visit_item_mut(self, node);
self.is_value = is_parent_value;
}

fn visit_table_mut(&mut self, node: &mut toml_edit::Table) {
Expand Down
105 changes: 105 additions & 0 deletions crates/toml/tests/testsuite/serde.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1115,3 +1115,108 @@ fn serialize_array_with_none_value() {
let err = toml::to_string(&input).unwrap_err();
snapbox::assert_eq("unsupported None value", err.to_string());
}

#[test]
fn serialize_array_with_optional_struct_field() {
#[derive(Debug, Deserialize, Serialize)]
struct Document {
values: Vec<OptionalField>,
}

#[derive(Debug, Deserialize, Serialize)]
struct OptionalField {
x: u8,
y: Option<u8>,
}

let input = Document {
values: vec![
OptionalField { x: 0, y: Some(4) },
OptionalField { x: 2, y: Some(5) },
OptionalField { x: 3, y: Some(7) },
],
};
let expected = "\
[[values]]
x = 0
y = 4
[[values]]
x = 2
y = 5
[[values]]
x = 3
y = 7
";
let raw = toml::to_string(&input).unwrap();
snapbox::assert_eq(expected, raw);

let input = Document {
values: vec![
OptionalField { x: 0, y: Some(4) },
OptionalField { x: 2, y: None },
OptionalField { x: 3, y: Some(7) },
],
};
let expected = "\
[[values]]
x = 0
y = 4
[[values]]
x = 2
[[values]]
x = 3
y = 7
";
let raw = toml::to_string(&input).unwrap();
snapbox::assert_eq(expected, raw);
}

#[test]
fn serialize_array_with_enum_of_optional_struct_field() {
#[derive(Debug, Deserialize, Serialize)]
struct Document {
values: Vec<Choice>,
}

#[derive(Debug, Deserialize, Serialize)]
enum Choice {
Optional(OptionalField),
Empty,
}

#[derive(Debug, Deserialize, Serialize)]
struct OptionalField {
x: u8,
y: Option<u8>,
}

let input = Document {
values: vec![
Choice::Optional(OptionalField { x: 0, y: Some(4) }),
Choice::Empty,
Choice::Optional(OptionalField { x: 2, y: Some(5) }),
Choice::Optional(OptionalField { x: 3, y: Some(7) }),
],
};
let expected = "values = [{ Optional = { x = 0, y = 4 } }, \"Empty\", { Optional = { x = 2, y = 5 } }, { Optional = { x = 3, y = 7 } }]
";
let raw = toml::to_string(&input).unwrap();
snapbox::assert_eq(expected, raw);

let input = Document {
values: vec![
Choice::Optional(OptionalField { x: 0, y: Some(4) }),
Choice::Empty,
Choice::Optional(OptionalField { x: 2, y: None }),
Choice::Optional(OptionalField { x: 3, y: Some(7) }),
],
};
let expected = "values = [{ Optional = { x = 0, y = 4 } }, \"Empty\", { Optional = { x = 2 } }, { Optional = { x = 3, y = 7 } }]
";
let raw = toml::to_string(&input).unwrap();
snapbox::assert_eq(expected, raw);
}

0 comments on commit 025e40e

Please sign in to comment.