Live SYSTEM_PROMPT on mDock had drifted heavily from the repo template
(detailed correspondent fuzzy-matching catalogue, full existing-names
list, refined title-generation rules). Reconciled by adopting the live
prompt as the new baseline in SYSTEM_PROMPT.txt and layering two fixes
on top:
1. Recipient rule (Rule 1): Matthias / Mathias Siebels and any address-
block variant ("Herr Siebels", "Empfaengeradresse Windscheidstr. 33")
must NEVER be set as correspondent — m is the recipient of nearly
every doc. Paul Siebels: also recipient by default, only correspondent
when nachweislich Autor (eigener Brief, Schadensmeldung von Paul).
Triggering misclassification (issue body): doc 280 (Vattenfall
Stromliefervertrag) was tagged correspondent="Matthias Siebels"
because the AI picked the recipient address block as sender.
2. Soften "Bevorzuge IMMER existierenden Correspondent" -> only when
semantic similarity is clear. Genuinely new senders (Versorger, Arzt,
Versicherer, Vermieter, ...) get a new correspondent rather than
being force-mapped to the nearest existing name. Fixes the
Vattenfall -> Telekom drift on docs 283/284 (also addressed by head
adding Vattenfall ID 257 manually).
Also migrated push_system_prompt.py from m/otto into this repo so the
deploy mechanism (render template -> push to /app/data/.env -> restart
paperless-ai) lives next to the template. Added RECIPIENT_EXCLUDE
filter so Matthias/Mathias Siebels are stripped from the rendered
correspondents list — defense in depth on top of the prompt rule.
Paperless correspondent records (IDs 3, 255) are preserved for the
historical doc assignments that still reference them.
Applied to live mDock paperless-ai (backup .env.bak.20260516T162255).
39 of 41 Siebels-correspondent doc assignments cleared + their
paperless-AI sqlite tracker rows (processed_documents,
history_documents, openai_metrics) deleted so they reclassify on the
next scan. Two kept (doc 117 Vollmacht from Paul, doc 130
Schadensmeldung filled by Paul — both genuine Paul-as-author cases per
the new rule).
Refs: m/mDMS#3
Two changes:
1. Migrate mover from m/otto (commit 9974937, otto#438) into this repo
at infra/mdms-mover/. mover.sh, mdms-mover.service, mdms-mover.timer,
README.md. Matches the live deployment on mDock byte-for-byte (modulo
the strip step below).
2. Add blank-page stripping before the inbox → toprocess promotion. A
page is dropped iff its embedded text is empty AND its rendered
thumbnail is >= MDMS_BLANK_THRESHOLD near-white pixels (default 0.97
per issue #2). Detects the empty backside of patch-T separator
sheets in duplex scans (mDMS#2).
strip_blank_pages.py uses PyMuPDF as the only Python dep — single
self-contained wheel, no `poppler-utils` apt-install on mdock. Mirrors
the uv-inline-deps single-file pattern of infra/paperless/generate_separator.py.
Edge cases:
- 1-page input: strip skipped entirely.
- All pages would drop: script exits 2, mover keeps file in inbox and
logs WARNING (no empty doc reaches Paperless).
- Strip script errors: mover falls back to plain mv, no scan blocked.
- MDMS_STRIP_BLANK=false: bypass strip entirely (emergency disable).
Deploy: rsync uv binary to mdock ~/.local/bin/uv (single static binary,
user-space, no apt), scp script + units, systemctl --user daemon-reload.
Verified live with synthetic 4-page (2 real + 1 blank + 1 real → 3
pages), 1-page (unchanged), all-blank (kept in inbox + warning) test
PDFs. Timer fires every ~70s as before.
Spun out mDMS strategy + tooling from m/otto into its own repo on 2026-05-15.
Migrated:
- docs/strategy.md (was: m/otto:docs/mdms-strategy.md)
- infra/paperless/ (config + audit + migrate scripts)
- infra/samba-canon/ (Canon MB5100 SMB1 bridge container)
History in m/otto: issues #429–#438. Going forward, all mDMS issues
file here. Sibling m/paperless (separate repo) remains the bare
Docker Compose for Paperless-ngx itself.