use crate::document::Document;
#[test]
fn top_level_comments_round_trip() {
let src =
"---\nQUILL: q\n# recipient's full name\nrecipient: Jane\nauthor: Alice\n---\n\nBody.\n";
let doc = Document::from_markdown(src).unwrap();
let emitted = doc.to_markdown();
assert!(
emitted.contains("# recipient's full name"),
"top-level YAML comment must survive round-trip\nGot:\n{}",
emitted
);
let doc2 = Document::from_markdown(&emitted).unwrap();
assert_eq!(
doc2.main()
.frontmatter()
.get("recipient")
.and_then(|v| v.as_str()),
Some("Jane"),
);
let emitted2 = doc2.to_markdown();
assert_eq!(emitted, emitted2, "round-trip must be idempotent");
}
#[test]
fn trailing_comments_become_own_line_on_round_trip() {
let src = "---\nQUILL: q\ntitle: My Document # this is a comment\n---\n\nBody.\n";
let doc = Document::from_markdown(src).unwrap();
let emitted = doc.to_markdown();
assert!(
emitted.contains("# this is a comment"),
"trailing comment text must survive\nGot:\n{}",
emitted
);
assert!(
emitted.contains("title: \"My Document\"\n# this is a comment"),
"trailing comment must normalise to own-line on the next line\nGot:\n{}",
emitted
);
let doc2 = Document::from_markdown(&emitted).unwrap();
assert_eq!(
doc2.main()
.frontmatter()
.get("title")
.and_then(|v| v.as_str()),
Some("My Document"),
);
}
#[test]
fn custom_tags_lose_tag_but_keep_value() {
let src = "---\nQUILL: q\nmemo_from: !fill 2d lt example\n---\n";
let doc = Document::from_markdown(src).unwrap();
let fm = doc.main().frontmatter();
assert_eq!(
fm.get("memo_from").and_then(|v| v.as_str()),
Some("2d lt example"),
"string value must survive tag parsing"
);
assert!(fm.is_fill("memo_from"), "fill marker must be recorded");
let emitted = doc.to_markdown();
assert!(
emitted.contains("memo_from: !fill"),
"`!fill` tag must round-trip\nGot:\n{}",
emitted
);
let doc2 = Document::from_markdown(&emitted).unwrap();
assert!(
doc2.main().frontmatter().is_fill("memo_from"),
"fill marker must survive a full round-trip"
);
let src2 = "---\nQUILL: q\nmemo_from: !include value.txt\n---\n";
let out = Document::from_markdown_with_warnings(src2).unwrap();
assert!(
out.warnings
.iter()
.any(|w| w.code.as_deref() == Some("parse::unsupported_yaml_tag")),
"expected unsupported_yaml_tag warning; got: {:?}",
out.warnings
);
let emitted2 = out.document.to_markdown();
assert!(
!emitted2.contains("!include"),
"unknown tag must not re-appear on emit\nGot:\n{}",
emitted2
);
}
#[test]
fn fill_tag_bare_null_round_trip() {
let src = "---\nQUILL: q\nrecipient: !fill\n---\n";
let doc = Document::from_markdown(src).unwrap();
let fm = doc.main().frontmatter();
assert!(fm.get("recipient").map(|v| v.is_null()).unwrap_or(false));
assert!(fm.is_fill("recipient"));
let emitted = doc.to_markdown();
assert!(
emitted.contains("recipient: !fill\n"),
"bare `!fill` must round-trip as `key: !fill`\nGot:\n{}",
emitted
);
}
#[test]
fn fill_tag_block_sequence_round_trip() {
let src = "---\nQUILL: q\nrecipient: !fill\n - Dr. Who\n - 1 TARDIS Lane\n---\n";
let doc = Document::from_markdown(src).unwrap();
let fm = doc.main().frontmatter();
assert!(fm.is_fill("recipient"));
let arr = fm.get("recipient").and_then(|v| v.as_array()).unwrap();
assert_eq!(arr.len(), 2);
assert_eq!(arr[0].as_str(), Some("Dr. Who"));
assert_eq!(arr[1].as_str(), Some("1 TARDIS Lane"));
let emitted = doc.to_markdown();
assert!(
emitted.contains("recipient: !fill\n"),
"`!fill` on sequence must emit `key: !fill` before the block\nGot:\n{}",
emitted
);
let doc2 = Document::from_markdown(&emitted).unwrap();
assert!(doc2.main().frontmatter().is_fill("recipient"));
assert_eq!(doc2, doc, "full round-trip must be equal");
}
#[test]
fn fill_tag_flow_sequence_round_trip() {
let src = "---\nQUILL: q\ntags: !fill [a, b, c]\n---\n";
let doc = Document::from_markdown(src).unwrap();
let fm = doc.main().frontmatter();
assert!(fm.is_fill("tags"));
assert_eq!(
fm.get("tags").and_then(|v| v.as_array()).map(|a| a.len()),
Some(3)
);
let emitted = doc.to_markdown();
let doc2 = Document::from_markdown(&emitted).unwrap();
assert!(doc2.main().frontmatter().is_fill("tags"));
assert_eq!(doc2, doc);
}
#[test]
fn fill_tag_empty_sequence_round_trip() {
let src = "---\nQUILL: q\nitems: !fill []\n---\n";
let doc = Document::from_markdown(src).unwrap();
let fm = doc.main().frontmatter();
assert!(fm.is_fill("items"));
assert_eq!(
fm.get("items").and_then(|v| v.as_array()).map(|a| a.len()),
Some(0)
);
let emitted = doc.to_markdown();
assert!(
emitted.contains("items: !fill []\n"),
"empty fill-sequence must round-trip as `key: !fill []`\nGot:\n{}",
emitted
);
let doc2 = Document::from_markdown(&emitted).unwrap();
assert_eq!(doc2, doc);
}
#[test]
fn fill_tag_mapping_rejected() {
let src = "---\nQUILL: q\nx: !fill {a: 1}\n---\n";
let err = Document::from_markdown(src).unwrap_err();
assert!(
err.to_string().contains("!fill") && err.to_string().contains("mapping"),
"expected mapping-rejection error; got: {}",
err
);
}
#[test]
fn fill_tag_all_scalar_types_round_trip() {
let src = concat!(
"---\nQUILL: q\n",
"s: !fill hello\n",
"i: !fill 42\n",
"f: !fill 3.14\n",
"b: !fill true\n",
"n: !fill\n",
"---\n",
);
let doc = Document::from_markdown(src).unwrap();
let fm = doc.main().frontmatter();
assert_eq!(fm.get("s").and_then(|v| v.as_str()), Some("hello"));
assert_eq!(fm.get("i").and_then(|v| v.as_i64()), Some(42));
assert_eq!(fm.get("f").and_then(|v| v.as_f64()), Some(3.14));
assert_eq!(fm.get("b").and_then(|v| v.as_bool()), Some(true));
assert!(fm.get("n").map(|v| v.is_null()).unwrap_or(false));
for key in ["s", "i", "f", "b", "n"] {
assert!(fm.is_fill(key), "{} must be fill-tagged", key);
}
let emitted = doc.to_markdown();
let doc2 = Document::from_markdown(&emitted).unwrap();
for key in ["s", "i", "f", "b", "n"] {
assert!(
doc2.main().frontmatter().is_fill(key),
"{} must remain fill-tagged after round-trip",
key
);
}
}
#[test]
fn original_quoting_style_is_not_preserved() {
let src = "---\nQUILL: q\nsingle_q: 'hello'\nunquoted: world\ndouble_q: \"already\"\n---\n";
let doc = Document::from_markdown(src).unwrap();
let emitted = doc.to_markdown();
assert!(
!emitted.contains("'hello'"),
"single-quoted string must not be re-emitted single-quoted\nGot:\n{}",
emitted
);
assert!(
emitted.contains("\"hello\""),
"single-quoted string must be re-emitted double-quoted\nGot:\n{}",
emitted
);
assert!(
emitted.contains("\"world\""),
"unquoted string must be re-emitted double-quoted\nGot:\n{}",
emitted
);
assert!(
emitted.contains("\"already\""),
"double-quoted string must survive as double-quoted\nGot:\n{}",
emitted
);
let doc2 = Document::from_markdown(&emitted).unwrap();
assert_eq!(
doc2.main()
.frontmatter()
.get("single_q")
.and_then(|v| v.as_str()),
Some("hello")
);
assert_eq!(
doc2.main()
.frontmatter()
.get("unquoted")
.and_then(|v| v.as_str()),
Some("world")
);
assert_eq!(
doc2.main()
.frontmatter()
.get("double_q")
.and_then(|v| v.as_str()),
Some("already")
);
}
#[test]
fn nested_sequence_comments_round_trip() {
let src =
"---\nQUILL: q\nitems:\n # before-first\n - a\n # between\n - b\n # after-last\n---\n";
let out = Document::from_markdown_with_warnings(src).unwrap();
assert!(
!out.warnings
.iter()
.any(|w| w.code.as_deref() == Some("parse::comments_in_nested_yaml_dropped")),
"no dropped-comment warning expected; nested comments are now preserved"
);
let emitted = out.document.to_markdown();
assert!(
emitted.contains("# before-first"),
"leading nested comment must round-trip\nGot:\n{}",
emitted
);
assert!(
emitted.contains("# between"),
"between-items nested comment must round-trip\nGot:\n{}",
emitted
);
assert!(
emitted.contains("# after-last"),
"trailing nested comment must round-trip\nGot:\n{}",
emitted
);
let doc2 = Document::from_markdown(&emitted).unwrap();
let emitted2 = doc2.to_markdown();
assert_eq!(emitted, emitted2, "round-trip must be idempotent");
}
#[test]
fn nested_mapping_comments_round_trip() {
let src = "---\nQUILL: q\nouter:\n # leading\n inner: 1\n # trailing\n---\n";
let doc = Document::from_markdown(src).unwrap();
let emitted = doc.to_markdown();
assert!(
emitted.contains("# leading"),
"leading nested mapping comment must round-trip\nGot:\n{}",
emitted
);
assert!(
emitted.contains("# trailing"),
"trailing nested mapping comment must round-trip\nGot:\n{}",
emitted
);
let doc2 = Document::from_markdown(&emitted).unwrap();
let emitted2 = doc2.to_markdown();
assert_eq!(emitted, emitted2);
}
#[test]
fn trailing_nested_comments_become_own_line() {
let src = "---\nQUILL: q\nitems:\n - a # inline\n - b\n---\n";
let doc = Document::from_markdown(src).unwrap();
let emitted = doc.to_markdown();
assert!(
emitted.contains("# inline"),
"trailing nested comment must survive\nGot:\n{}",
emitted
);
assert!(
!emitted.contains("\"a\" # inline"),
"trailing nested comment must normalise to own-line\nGot:\n{}",
emitted
);
}