pdftract/tests/error_recovery/fixtures/gen_missing_mediabox.py
jedarden 4d6fd8a4ab test(pdftract-4w0v4): implement adversarial test corpus + integration harness
Add 7 adversarial PDF fixtures exercising Phase 1 error-recovery paths:
- xref_30pct_bad_offsets.pdf: 100 objects, 30 bad xref offsets
- missing_mediabox_all_pages.pdf: 10 pages, no /MediaBox at any level
- missing_endobj.pdf: object 5 missing endobj marker
- truncated_mid_stream.pdf: FlateDecode stream truncated mid-decompression
- int_overflow_bbox.pdf: /BBox value 99999999999999999 (i32 overflow)
- nested_failure.pdf: every page has at least one diagnostic
- combined_failures.pdf: combines multiple failure modes (keystone INV-8 test)

Each fixture has a sibling .expected_diagnostics.json file with threshold
counts (>= not == per EC-07/EC-09 to tolerate drift).

Integration test harness (error_recovery_integration.rs):
- assert_diagnostic_count_at_least() helper for threshold checking
- assert_no_panic() helper using std::panic::catch_unwind for INV-8
- Individual test functions for each fixture
- Cumulative test_inv_8_no_panics_across_all_fixtures()

All 8 tests pass. INV-8 verified: zero panics across all fixtures.

Closes: pdftract-4w0v4
2026-05-25 14:30:24 -04:00

58 lines
1.2 KiB
Python

#!/usr/bin/env python3
"""Generate missing_mediabox_all_pages.pdf - a 10-page PDF with NO /MediaBox at any level."""
pages = []
for i in range(10):
pages.append(f"{3+i} 0 obj\n<< /Type /Page /Parent 2 0 R /Contents 13 0 R /Resources << /Font << /F1 14 0 R >> >> >>\nendobj\n")
pages_joined = ''.join(pages)
kids = ' '.join([f'{3+i} 0 R' for i in range(10)])
PDF_CONTENT = f"""%PDF-1.4
1 0 obj
<< /Type /Catalog /Pages 2 0 R >>
endobj
2 0 obj
<< /Type /Pages /Kids [{kids}] /Count 10 >>
endobj
{pages_joined}13 0 obj
<< /Length 44 >>
stream
BT
/F1 12 Tf
100 700 Td
(Test) Tj
ET
endstream
endobj
14 0 obj
<< /Type /Font /Subtype /Type1 /BaseFont /Helvetica >>
endobj
xref
0 15
0000000000 65535 f
0000000009 00000 n
0000000058 00000 n
0000000135 00000 n
0000000248 00000 n
0000000361 00000 n
0000000474 00000 n
0000000587 00000 n
0000000700 00000 n
0000000813 00000 n
0000000926 00000 n
0000001039 00000 n
0000001152 00000 n
0000001265 00000 n
trailer
<< /Size 15 /Root 1 0 R >>
startxref
1355
%%EOF
"""
with open('missing_mediabox_all_pages.pdf', 'wb') as f:
f.write(PDF_CONTENT.encode('latin-1'))
print("Generated missing_mediabox_all_pages.pdf")
print("All 10 pages are missing /MediaBox - should default to 612x792 letter size")