use crate::document::Document;
fn assert_round_trip(label: &str, src: &str) {
let a = Document::from_markdown(src)
.unwrap_or_else(|e| panic!("{}: from_markdown failed on original: {}", label, e));
let emitted = a.to_markdown();
let b = Document::from_markdown(&emitted).unwrap_or_else(|e| {
panic!(
"{}: from_markdown failed on emitted document.\nError: {}\nEmitted:\n{}",
label, e, emitted
)
});
assert_eq!(
a, b,
"{}: round-trip produced different Documents.\nEmitted:\n{}",
label, emitted
);
}
#[test]
fn fixture_corpus_round_trip() {
let manifest_dir = env!("CARGO_MANIFEST_DIR");
let quills_dir = std::path::Path::new(manifest_dir)
.join("..") .join("fixtures")
.join("resources")
.join("quills");
let mut fixture_paths: Vec<std::path::PathBuf> = Vec::new();
if let Ok(entries) = std::fs::read_dir(&quills_dir) {
for entry in entries.flatten() {
if entry.path().is_dir() {
collect_md_files(&entry.path(), &mut fixture_paths);
}
}
}
let resources_dir = quills_dir.parent().unwrap();
for entry in std::fs::read_dir(resources_dir).unwrap().flatten() {
let path = entry.path();
if path.extension().and_then(|e| e.to_str()) == Some("md") {
fixture_paths.push(path);
}
}
let appreciated = resources_dir
.join("appreciated_letter")
.join("appreciated_letter.md");
if appreciated.exists() {
fixture_paths.push(appreciated);
}
assert!(
!fixture_paths.is_empty(),
"no fixture files found — check paths"
);
let mut passed = 0usize;
let mut skipped = 0usize;
let mut failed = 0usize;
let mut failures: Vec<String> = Vec::new();
for path in &fixture_paths {
let label = path.to_string_lossy();
let src = match std::fs::read_to_string(path) {
Ok(s) => s,
Err(e) => {
eprintln!("SKIP {}: cannot read: {}", label, e);
skipped += 1;
continue;
}
};
match Document::from_markdown(&src) {
Err(_) => {
skipped += 1;
continue;
}
Ok(a) => {
let emitted = a.to_markdown();
match Document::from_markdown(&emitted) {
Err(e) => {
failed += 1;
failures.push(format!(
"FAIL {}: re-parse failed: {}\nEmitted:\n{}",
label, e, emitted
));
}
Ok(b) => {
if a == b {
passed += 1;
} else {
failed += 1;
failures.push(format!(
"FAIL {}: documents differ after round-trip.\nEmitted:\n{}",
label, emitted
));
}
}
}
}
}
}
if !failures.is_empty() {
panic!(
"Fixture round-trip failures ({} failed, {} passed, {} skipped):\n{}",
failed,
passed,
skipped,
failures.join("\n\n")
);
}
assert!(
passed > 0,
"No fixtures passed round-trip — did all files get skipped?"
);
eprintln!(
"fixture_corpus_round_trip: {} passed, {} skipped",
passed, skipped
);
}
fn collect_md_files(dir: &std::path::Path, out: &mut Vec<std::path::PathBuf>) {
if let Ok(entries) = std::fs::read_dir(dir) {
for entry in entries.flatten() {
let path = entry.path();
if path.is_dir() {
collect_md_files(&path, out);
} else if path.extension().and_then(|e| e.to_str()) == Some("md") {
out.push(path);
}
}
}
}
#[test]
fn emit_twice_is_byte_equal() {
let src = "\
---
QUILL: test@1.0.0
title: Stability Test
flags:
- on
- 'yes'
- null
count: 42
nested:
key: value
---
Body content here.
";
let doc = Document::from_markdown(src).unwrap();
let first = doc.to_markdown();
let second = doc.to_markdown();
assert_eq!(
first, second,
"to_markdown must be deterministic (byte-equal on repeated calls)"
);
}
#[test]
fn round_trip_booleans() {
let src = "---\nQUILL: q\nflag_true: true\nflag_false: false\n---\n";
assert_round_trip("booleans", src);
}
#[test]
fn round_trip_null() {
let src = "---\nQUILL: q\nnull_field: null\n---\n";
assert_round_trip("null", src);
}
#[test]
fn round_trip_numbers() {
let src = "---\nQUILL: q\ncount: 42\nfloat: 3.14\n---\n";
assert_round_trip("numbers", src);
}
#[test]
fn round_trip_string_ambiguous() {
let src = "---\nQUILL: q\nfield_on: \"on\"\nfield_yes: \"yes\"\nfield_01234: \"01234\"\n---\n";
assert_round_trip("ambiguous strings", src);
}
#[test]
fn round_trip_nested_map() {
let src = "---\nQUILL: q\nsender:\n name: Alice\n city: Springfield\n---\n";
assert_round_trip("nested map", src);
}
#[test]
fn round_trip_sequence() {
let src = "---\nQUILL: q\ntags:\n - demo\n - test\n---\n";
assert_round_trip("sequence", src);
}
#[test]
fn round_trip_empty_sequence() {
let src = "---\nQUILL: q\nempty: []\n---\n";
assert_round_trip("empty sequence", src);
}
#[test]
fn round_trip_cards() {
let src = "\
---
QUILL: q
title: Test
---
Body text.
---
CARD: section
heading: Chapter 1
---
Card body here.
";
assert_round_trip("cards", src);
}
#[test]
fn round_trip_card_empty_body() {
let src = "\
---
QUILL: q
title: Test
---
---
CARD: empty_body_card
title: No body
---
";
assert_round_trip("card with empty body", src);
}
#[test]
fn round_trip_string_with_escapes() {
let src = "---\nQUILL: q\npath: \"C:\\\\Users\\\\test\"\n---\n";
assert_round_trip("string with backslash", src);
}
#[test]
fn round_trip_multiline_string() {
let src = "---\nQUILL: q\nbio: \"Line one\\nLine two\"\n---\n";
assert_round_trip("multiline string", src);
}
#[test]
fn round_trip_quill_version_selectors() {
for qref in &["q", "q@1", "q@1.2", "q@1.2.3", "q@latest"] {
let src = format!("---\nQUILL: {}\ntitle: t\n---\n", qref);
assert_round_trip(&format!("quill ref {}", qref), &src);
}
}
#[test]
fn empty_map_omitted_from_emit() {
use crate::value::QuillValue;
use indexmap::IndexMap;
let mut frontmatter: IndexMap<String, QuillValue> = IndexMap::new();
frontmatter.insert(
"empty_obj".to_string(),
QuillValue::from_json(serde_json::json!({})),
);
frontmatter.insert(
"real_field".to_string(),
QuillValue::from_json(serde_json::json!("hello")),
);
use crate::document::{Card, Frontmatter, Sentinel};
use crate::version::{QuillReference, VersionSelector};
let main = Card::new_with_sentinel(
Sentinel::Main(QuillReference::new(
"test".to_string(),
VersionSelector::Latest,
)),
Frontmatter::from_index_map(frontmatter),
String::new(),
);
let doc = crate::document::Document::from_main_and_cards(main, vec![], vec![]);
let md = doc.to_markdown();
assert!(
!md.contains("empty_obj"),
"empty object should be omitted from emit, got:\n{}",
md
);
assert!(
md.contains("\"hello\""),
"real field should appear double-quoted, got:\n{}",
md
);
}